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写 在 前 面 的 话 


我 的 兄 第 Todd 目 前 正在 进行 从 硬件 到 编程 领域 的 工作 转弯。 我 曾 提醒 他 
下 一 次 大 章 命 的 重点 将 是 遗传 工程 。 


本 书 由 “ 行 行 ? 整 理 ， 如 果 你 不 知道 读 什么 书 或 者 想 获 得 更 多 免费 电子 书 
请 加 小 编 微 信 或 QQ: 2338856113 小 编 也 和 结交 一 些 喜 欢 读 书 的 朋友 或 
者 关注 小 编 个 人 微 信 公众 号 名 称 : 幸福 的 味道 id: d716-716 为 了 方便 书 
友 朋 友 找 书 和 看 书 ， 小 编 自 己 做 了 一 个 电子 书 下 载 网 站 ， 网 站 的 名 称 
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我 们 的 微生物 技术 将 能 制造 食品 、 燃 油 和 塑料 ;它们 都 是 清洁 的 ， 不 会 
造成 污染 ， 而 且 能 使 人 类 进一步 透视 物理 世界 的 奥秘 。 我 认为 相 比 之 下 
电脑 的 进步 会 显得 微不足道 。 


但 随后 ， 我 又 意识 到 目 己 正 在 犯 一 些 科 弥 作家 名 犯 的 错误 : 在 技术 中 迷 
失 了 这 种 事情 在 科幻 小 说 里 第 有 发 生 ) ! 如 果 是 一 名 有 经 验 的 作家 ， 

就 知道 绝对 不 能 就 事 论 事 ， 必 须 以 人 为 中 心 。 遗 传 对 我 们 的 生命 有 非常 
大 的 影响 ， 但 不 能 十 分 确定 它 能 抹 淡 计算 机 间 命 一 一 或 至 少 信 息 革 命 

一 一 的 影响 。 信 息 涉 及 人 相互 间 的 沟通 : 的 确 ， 汽 车 和 轮子 的 友 明 都 非 
常 重要 ， 但 它们 最 终 亦 如 此 而 已 。 真 正 重 要 的 还 是 我 们 与 世界 的 关系 ， 
而 其 中 最 关键 的 就 是 通信 。 


这 本 书 或 许 能 说 明 一 些 问 题 。 许 多 人 认为 我 有 点 儿 大 胆 或 者 稍微 有 些 狂 
妄 ， 居 然 把 所 有 家 当 都 摆 到 了 Web 上 。“ 这 样 做 还 有 谁 来 买 它 呢 ? ”他 们 
问 。 假 如 我 是 一 个 十 分 守旧 的 人 ， 那 么 绝对 不 这 样 干 。 但 我 确实 不 想 再 
沿 原 来 的 老路 再 写 一 本 计算 机 参考 书 了。 我 不 知道 最 终 会 及 生 什 么 事 
情 ， 但 的 确认 为 这 是 我 对 一 本 书 作出 的 最 明智 的 一 个 决定 。 


至 少 有 一 件 事 是 可 以 肯定 的 ， 人 人 们 开始 向 我 发 送 纠 错 反 馈 。 这 是 一 个 令 
人 震惊 的 体验 ， 因 为 读者 会 看 到 书 中 的 每 一 个 角落 ， 并 揪 出 那些 藏匿 得 
很 深 的 技术 及 语法 错误 。 这 样 一 来 ， 和 其 他 以 传统 方式 发 行 的 书 不 同 ， 
我 就 能 及 时 改正 已 知 的 所 有 类 别 的 错误 ， 而 不 是 让 它们 最 终 印 成 铅字 ， 
笛 而 星之 地 出 现在 各 位 的 面前 。 俗 话说 , “当局 者 迷 ， 和 劳 观 者 清 ?。 人 们 
对 书 中 的 错误 是 非常 敏感 的 ， 往 往 坚 不 客气 地 指出 : “我 想 这 样 说 是 错 
误 的 ， 我 的 看 法 是 .…….”。 在 我 仔细 研究 后 ， 往 往 发 现 自己 确实 有 不 当 














之 处 ， 而 这 是 当初 写作 时 根本 没有 意识 到 的 《检查 多 少 近 也 不 行 ) 。 我 
意识 到 这 和 古 群 体力 量 的 一 个 可 喜 的 反映 ， 它 使 这 本 书 显 得 的 确 与 众 不 
同 。 


但 我 随 之 又 听 到 了 另 一 个 声音 :“ 好 吧 ， 你 在 那儿 放 的 电子 版 的 确 很 有 
创意 ， 但 我 想 要 的 是 从 真正 的 出 版 社 那 里 印刷 的 一 个 版 本 ! ”事实 上 ， 

我 作出 了 许多 努力 ， 证 它 用 普通 打印 机 机 就 能 得 到 很 好 的 阅读 效果 ， 但 
仍然 不 象 真 正印 刷 的 书 那样 正规 。 许 多 人 不 想 在 屏幕 上 看 完整 本 书 ， 也 
不 喜欢 拿 着 一 登 纸 阅读 。 无 论 打 印 格式 有 多 么 好 ， 这 些 人 喜欢 是 仍然 是 
真正 的 “ 书 ”(〈 激 光 打 印 机 的 墨盒 也 太 贵 了 一 点 ) 。 现 在 看 来 ， 计 算 机 的 
革命 仍 未 使 出 版 界 完 全 走出 传统 的 模式 。 但 是 ， 有 一 个 学 生 癌 我 推荐 了 
未 来 出 版 的 一 种 模式 ， 书 籍 将 首先 在 互联 网 上 出 版 ， 然 后 只 有 在 绝对 必 
要 的 前 提 下 ， 才 会 印刷 到 纸张 上 。 目 前 ， 为 数 众多 的 书籍 销售 都 不 十 分 
理想 ， 许 多 出 版 社 都 在 亏本 。 但 如 采用 这 种 方式 出 版 ， 就 显得 灵活 得 

多 ， 也 更 容易 你 证 恬 利 。 


这 本 书 也 从 妃 一 个 角度 也 给 了 我 深刻 的 局 迪 。 我 刚 开始 的 时 候 以 为 
Java“ 只 是 另 一 种 程序 设计 语言 >。 这 个 想法 在 许多 情况 下 都 是 成 立 的 。 
但 随 独 时 间 的 推移 ， 我 对 它 的 学 习 也 您 加深 入， 开始 意识 到 它 的 基本 宗 
则 与 我 见 过 的 其 他 所 有 语言 都 有 所 区 别 。 


程序 设计 与 对 复杂 性 的 操控 有 很 大 的 关系 : 对 一 个 准备 解决 的 问题 ， 它 
的 复杂 程度 取决 用 于 解决 它 的 机 器 的 复杂 程度 。 正 是 由 于 这 一 复杂 性 的 
存在 ， 我 们 的 程序 设计 项 目 屡屡 失败 。 对 于 我 以 前 接触 过 的 所 有 编程 语 
言 ， 它 们 都 没 能 跳 过 这 一 框框 ， 由 此 决定 了 它们 的 主要 设计 目标 就 是 克 
服 程序 开发 与 维护 中 的 复杂 性 。 当 然 ， 许 多 语言 在 设计 时 就 已 考虑 到 了 
复杂 性 的 问题 。 但 从 另 一 角度 看 ， 实 际 设计 时 肯定 会 有 男 一 些 问题 浮现 
出 来 ， 需 把 它们 考虑 到 这 个 复杂 性 的 问题 里 。 不 可 避免 地 ， 其 他 那些 问 
题 最 后 会 变 成 最 让 程序 员 头 痛 的 。 例 如 ，C++ 必 须 同 C 保 持 向 后 兼容 
(使 C 程 序 员 能 尽快 地 适应 新 环境 ) ， 同 时 又 要 保证 编程 的 效率 。 

C++ 在 这 两 个 方面 都 设计 得 很 好 ， 为 其 赢得 了 不 少 的 声誉 。 但 它们 同时 
也 暴露 出 了 额外 的 复杂 性 ， 阻 碍 了 某 些 项 目的 成 功 实现 〈 当 然 ， 你 可 以 
责备 程序 员 和 管理 层 ， 但 假如 一 种 语言 能 通过 捕获 你 的 错误 而 提供 帮 
助 ， 它 为 什么 不 那样 做 昵 ? ) 。 作 为 另 一 个 例子 ，Visual Basic (VB) 
同 当初 的 BASIC 有 关 的 紧密 的 联系 。 而 BASIC 并 没有 打算 设计 成 一 种 能 
全 面 解决 问题 的 语言 ， 所 以 堆 加 到 VB 身上 的 所 有 扩展 都 造成 了 令 人 头 
痛 和 难于 管理 和 维护 的 语法 。 男 一 方面 ，C++、VB 和 其 他 如 Smalltalk 之 















































类 的 语言 均 在 复杂 性 的 问题 上 下 了 一 看 功夫 。 由 此 得 到 的 结果 便 是 ， 它 
们 在 解决 特定 类 型 的 问题 时 是 非常 成 功 的 。 


在 理解 到 Java 最 终 的 目标 是 减 经 程序 员 的 负担 时 ， 我 才 真 正 感受 到 了 震 
憾 ， 尽 管 它 的 潜台词 好 象 是 说 ;“ 除 了 缩短 时 间 和 减 小 产生 健壮 代码 的 
难度 以 外 ， 我 们 不 关心 其 他 任何 事情 。” 在 目前 这 个 初级 阶段 ， 达 到 那 
个 目标 的 后 果 便 是 代码 不 能 特别 快 地 运行 (尽管 有 许多 保证 都 说 Java 终 
究 有 一 天 会 运行 得 多 么 快 ) ， 但 它 确实 将 开发 时 间 缩短 到 令 人 惊讶 的 地 
步 一 一 几乎 只 有 创建 一 个 等 效 C++ 程序 一 半 甚至 更 短 的 时 间 。 这 段 节省 
下 来 的 时 间 可 以 产生 更 大 的 效益 ， 但 Java 并 不 仅 止 于 此 。 它 甚至 更 上 一 
层 楼 ， 将 重要 性 越 来 越 明 显 的 一 切 复杂 任务 都 封装 在 内 ， 比 如 网 络 程序 
和 多 线程 处 理 等 等 。Java 的 各 种 语言 特性 和 库 在 任何 时 候 都 能 使 那些 任 
务 轻而易举 完成 。 而 且 最 后 ， 它 解决 了 一 些 真正 有 些 难度 的 复杂 问题 
跨 平台 程序 、 动 态 代码 改换 以 及 安全 保护 等 等 。 换 在 从 前 ， 其 中 任何 每 
一 个 都 能 使 你 头 大 如 斗 。 所 以 不 管 我 们 见 到 了 什么 性 能 问题 ，Java 的 保 
证 仍然 是 非常 有 效 的 ， 它 使 程序 员 显著 提高 了 程序 设计 的 效率 ! 


在 我 看 来 ， 编 程 效 率 提 升 后 影响 最 大 的 就 是 web。 网 络 程序 设计 以 前 非 
常 困 难 ， 而 Java 使 这 个 问题 迎刃而解 (而 且 Java 也 在 不 断 地 进步 ， 使 解 
决 这 类 问题 变 得 越 来 越 容易 ) 。 网 络 程序 的 设计 要 求 我 们 相互 间 更 有 效 
率 地 沟通 ， 而 且 至 少 要 比 电话 通信 来 得 便宜 〈 仅 仅 电 子 函 件 就 为 许多 公 
司 带 来 了 好 处 )。 随 着 我 们 网 上 通信 越 来 越 频 繁 ， 令 人 震惊 的 事情 会 慢 
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在 各 个 方面 : 创建 程序 ， 按 计划 编制 程序 ， 构 造 用 户 界面 ， 使 程序 能 与 
APA; 在 不 同类 型 的 机 器 上 运行 程序 ， 以 及 方便 地 编写 程序 ， 使 其 
能 通过 因特网 通信 一 一 Java 提 高 了 人 与 人 之 间 的 “通信 带宽 ?。 而 且 我 认 
为 通信 章 命 的 结果 可 能 并 不 单单 是 数量 庞大 的 比特 到 处 传 来 传 去 那么 简 
单 。 我 们 认为 认 清 真正 的 革命 发 生 在 哪里 ， 因 为 人 和 人 之 间 的 交流 变 得 
更 方便 了 一 一 个 体 与 个 体 之 间 ， 个 体 与 组 之 间 ， 组 与 组 之 间 ， 甚 至 在 星 
球 之 间 。 有 人 预言 下 一 次 大 革命 的 发 生 就 是 由 于 足够 多 的 人 和 足够 多 的 
相互 连接 造成 的 ， 而 这 种 革命 是 以 整个 世界 为 基础 友 生 的 。Java 可 能 
是 、 也 可 能 不 是 促成 那 次 革命 的 直接 因素 ， 但 我 在 这 里 至 少 感觉 目 己 在 
做 一 些 有 意义 的 工作 一 一 尝试 教会 大 家 一 种 重要 的 语言 ! 
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同人 类 任何 语言 一 样 ，Java 为 我 们 提供 了 一 种 表达 思想 的 方式 。 如 操作 
得 当 ， 同 其 他 方式 相 比 ， 随 大 问题 变 得 人 鳄 大 和 分 复 杂 ， 这 种 表达 方式 的 
方便 性 和 灵活 性 会 显露 无 遗 。 


不 可 将 Java 简 时 想象 成 一 系列 特性 的 集合 ; 如 孤立 地 看 ， 有 些 特性 是 没 
有 任何 意义 的 。 只 有 在 考虑 “设计 ”、 而 非 考 虑 简单 的 编码 时 ， 才 可 真正 
体会 到 Java 的 强大 。 为 了 按 这 种 方式 理解 Java， 首 先 必须 掌握 它 与 编程 
的 一 些 基 本 概念 。 本 书 讨论 了 编程 问题 、 它 们 为 何 会 成 为 问题 以 及 Java 
用 以 解决 它们 的 方法 。 所 以 ， 我 对 每 一 章 的 解释 都 建立 在 如 何 用 语言 解 
决 一 种 特定 类 型 的 问题 基础 上 。 按 这 种 方式 ， 我 希望 引导 您 一 步 一 步 地 
进入 Java 的 世界 ， 使 其 最 终 成 为 您 最 自然 的 一 种 语言 。 


贯穿 本 书 ， 我 试图 在 您 的 大 脑 里 建立 一 个 模型 一 一 或 者 说 一 个 “知识 结 
构 ”。 这 样 可 加 深 对 语言 的 理解 。 奉 遇 到 难 解 之 处 ， 应 学 会 把 它 填 入 这 
个 模型 的 对 应 地 方 ， 然 后 自行 演绎 出 答案 。 事 实 上 ， 学 习 任何 语言 时 ， 
脑海 里 有 一 个 现成 的 知识 结构 往往 会 起 到 事半功倍 的 效果 。 


1. 前 提 


本 书 假定 读者 对 编程 多 少 有 些 熟 悉 。 应 已 知道 程序 是 一 系列 语句 的 集 
合 ， 知 道子 程序 / 函数 / 宏 是 什么 ， 知 道 象 If" 这样 的 控制 语句 ， 也 知 
道 象 "while" 这 样 的 循环 结构 。 注 意 这 些 东西 在 大 量 语言 里 都 是 类 似 的 。 
假如 您 学 过 一 种 宏 语言 ， 或 者 用 过 Perl 之 类 的 工具 ， 那 么 它们 的 基本 概 
念 并 无 什么 区 别 。 总 之 ， 只 要 能 习惯 基本 的 编程 概念 ， 就 可 顺利 阅读 本 
书 。 当 然 ，C/C++ 程 序 员 在 阅读 时 能 占 到 更 多 的 便宜 。 但 即使 不 熟悉 
C， 一 样 不 要 把 目 己 排除 在 外 《尽管 以 后 的 学 习 要 付出 更 大 的 努力 ) 。 
我 会 讲述 面 癌 对 象 编 程 的 概念 ， 以 及 Java 的 基本 控制 机 制 ， 所 以 不 用 担 
心目 己 会 打 不 好 基础 。 况 且 ， 您 需要 学 习 的 第 一 类 知识 就 会 涉及 到 基本 
的 流程 控制 语句 。 


尽管 经 常 都 会 谈 及 C 和 C++ 语 言 的 一 些 特性 ， 但 并 没有 打算 使 它们 成 为 
内 部 参考 ， 而 是 想 帮 助 所 有 程序 员 都 能 正确 地 看 待 那 两 种 语言 。 毕 竟 ， 
Java 和 是 从 它们 那里 衍生 出 来 的 。 我 将 试 痢 尽 可 能 地 简化 这 些 引 用 和 参 
考 ， 并 合理 地 解释 一 名 非 C/C++ 程 序 员 通常 不 太 熟 悉 的 内 容 。 















































2. Java 的 学 习 


在 我 第 一 本 书 《Using ” C++》 面市 的 几乎 同一 时 间 (Osborne/McGraw- 
Hil 于 1989 年 出 版 )， 我 开始 教授 那 种 语言 。 程 序 设 计 语 言 的 教授 已 成 
为 我 的 专业 。 目 1989 年 以 来 ， 我 便 在 世界 各 地 见 过 许多 外 色欲 睡 、 满 脸 
荡然 以 及 困惑 不 解 的 面容 。 开 始 在 室内 面向 较 少 的 一 组 人 授课 以 后 ， 我 
从 作业 中 发 现 了 一 些 特别 的 问题 。 即 使 那些 上 课 面 带 会 心 的 微笑 或 者 频 
频 点 头 的 学 生 ， 对 许多 问题 也 存在 认识 上 的 混 消 。 在 过 去 几 年 间 的 “ 软 
件 开 发 会 议 ” 上 ， 由 我 主持 C++ 分 组 讨论 会 〈 现 在 变 成 了 Java 讨 论 会 ) 。 
有 的 演讲 人 试图 在 很 短 的 时 间 内 加 听众 灌输 过 多 的 主题 。 所 以 到 最 后 ， 
尽管 听众 的 水 平 都 还 可 以 ， 而 且 提 供 的 材料 也 很 充足 ， 但 仍然 损失 了 一 
部 分 听众 。 这 可 能 是 由 于 问 得 太 多 了 ， 但 由 于 我 是 那些 采取 传统 授课 方 
式 的 人 之 一 ， 所 以 很 想 使 每 个 人 都 能 跟 上 讲课 进度 。 


有 段 时 间 ， 我 编制 了 大 量 教学 简报 。 经 过 不 断 的 试验 和 修订 〈 或 称 * 反 
复 ”， 这 是 在 Java 程 序 设计 中 非常 有 用 的 一 项 技术 ) ， 最 后 成 功 地 在 一 
门 课程 中 集成 了 从 我 的 教学 经 验 中 总 结 出 来 的 所 有 东西 一 一 我 在 很 长 一 
段 时 间 里 都 在 使 用 。 其 中 由 一 系列 离散 的 、 易 于 消化 的 小 步骤 组 成 ， 而 
ARV) AE a ES SRS. RRA Java Fit = 
上 公布 了 这 一 课程 ， 大 家 可 到 http:/www.BruceEckel.com 了 解 详 情 〈 对 
的 介绍 也 以 CD-ROM 的 形式 提供 ， 具 体 信息 可 在 同样 的 Web 站 点 
LF) 。 


从 每 一 次 研讨 会 收 到 的 反馈 都 帮助 我 修改 及 重新 制订 学 习 材料 的 重心 ， 

直到 我 最 后 认为 它 成 为 一 个 完善 的 教学 载体 为 止 。 但 本 书 并 非 仅 仅 古 一 
本 教科 书 一 一 我 尝试 在 其 中 六 入 尽 可 能 多 的 信息 ， 并 按照 主题 进行 了 有 
序 的 分 类 。 无 论 如 何 ， 这 本 书 的 主要 宗旨 是 为 那些 独立 学 习 的 人 士 服 

务 ， 他 们 正 准备 深入 一 门 新 的 程序 设计 语言 ， 而 没有 太 大 的 可 能 参加 此 
类 专业 研讨 会 。 


3. 目标 


就 象 我 的 前 一 本 书 《Thinking in C++》 一 样 ， 这 本 书面 向 语言 的 教授 进 
行 了 民 好 的 结构 与 组 织 。 特 别 地 ， 我 的 目标 是 建立 一 套 有 序 的 机 制 ， 可 
帮助 我 在 上 自己 的 研讨 会 上 更 好 地 进行 语言 教学 。 在 我 思考 书 中 的 一 章 
时 ， 实 际 上 是 在 想 如 何 教 好 一 笛 课 。 我 的 目标 是 得 到 一 系列 规模 适中 的 
教学 模块 ， 可 以 在 合理 的 时 间 内 教 完 。 随 后 是 一 些 精心 挑选 的 练习 ， 可 
以 在 课堂 上 当即 完成 。 




















在 这 本 书 中 ， 我 想 达 到 的 目标 总 结 如 下 : 


O 每 一 次 都 将 教学 内 容 向 前 推进 一 小 步 ， 便 于 读者 在 继续 后 面 的 学 习 
前 消化 前 面 的 内 容 。 


(2) “采用 的 示例 尽 可 能 简短 。 当 然 ， 这 样 做 有 时 会 妨碍 我 解决 < 现实 世 
界 ” 的 问题 。 但 我 同时 也 发 现 对 那些 新 手 来 说 ， 如 果 他 们 能 理解 每 一 个 
细节 ， 那 么 一 般 会 产生 更 大 的 学 习 兴 趣 。 而 假如 他 们 一 开始 就 修 要 解决 
的 问题 的 深度 和 广度 所 震 尺 ， 那 么 一 般 都 不 会 收 到 很 好 的 学 习 效 果 。 男 























外 在 实际 教学 过 程 中 ， 对 能 够 摘录 的 代码 数量 是 有 严重 限制 的 。 男 一 方 
面 ， 这 样 做 无 疑 会 有 些 人 会 批评 我 及 用 了 “不 真实 的 例子 ”， 但 只 要 能 起 


到 良好 的 效果 ， 我 宁愿 接受 这 一 指 贡 。 


(3) 要 揭示 的 特性 按照 我 精心 挑选 的 顺序 依次 出 场 ， 而 且 尽 可 能 符合 读 
者 的 思想 历程 。 当 然 ， 我 不 可 能 永远 都 做 到 这 一 点 ; 在 那些 情况 下 ， 会 
给 出 一 段 简要 的 声明 ， 指 出 这 个 问题 。 


(4) 只 把 我 认为 有 助 于 理解 语言 的 东西 介绍 给 读者 ， 而 不 是 把 我 知道 的 
一 切 东西 都 拌 出 来 ， 这 并 非 藏 私 。 我 认为 信息 的 重要 程度 是 存在 一 个 合 
理 的 层次 的 。 有 些 情况 是 95% 的 程序 员 都 永远 不 必 了 解 的 。 如 强行 学 
习 ， 只 会 干扰 他 们 的 正常 思维 ， 从 而 加 深 语 言 在 他 们 面前 表现 出 来 的 难 
度 。 以 C 语 言 为 例 ， 假 如 你 能 记 住 运算 符 优先 次 序 表 (我 从 来 记 不 

住 )， 那 么 束 可 以 写 出 更 “聪明 ”的 代码 。 但 再 深入 想 一 层 ， 那 也 会 使 代 
eo pe emer 

号 即 可 。 


(5) 每 一 市 都 有 明确 的 学 习 重 点 ， 所 以 教学 时 间 (以 及 练习 的 间隔 时 
间 ) 非常 短 。 这 样 做 不 仅 能 保持 读者 思想 的 活跃 ， 也 能 使 问题 更 容易 理 
解 ， 对 目 己 的 学 习 产 生 更 大 的 信心 。 


(6) ”提供 一 个 坚实 的 基础 ， 使 读者 能 充分 理解 问题 ， 以 便 更 容易 转 问 一 
些 更 加 困难 的 课程 和 书籍 。 


4. 联机 文档 
由 Sun 微 系统 公司 提供 的 Java 语 言 和 库 〈 可 免费 下 载 ) 配套 提供 了 电子 


版 的 用 户 帮 助手 册 ， 可 用 Web 浏 览 占 阅读 。 此 外 ， 由 其 他 三 商 开 发 的 几 
乎 所 有 类 似 产 品 都 有 一 套 等 价 的 文档 系统 。 而 目前 出 版 的 与 Java 有 关 的 




















几乎 所 有 书籍 都 重复 了 这 份 文档 。 所 以 你 要 么 已 经 拥有 了 它 ， 要 么 需要 
下 载 。 所 以 除非 特别 必要 ， 人 否则 本 书 不 会 重复 那 份 文档 的 内 容 。 因 为 一 
般 地 说 ， 用 Web 浏 览 圳 得 找 与 类 有 关 的 资料 比 在 书 中 得 找 方便 得 多 《〈 电 
子 版 的 东西 更 新 也 快 ) 。 只 有 在 需要 对 文档 进行 补充 ， 以 便 你 能 理解 一 
个 特定 的 例子 时 ， 本 书 才 会 提供 有 关 类 的 一 些 附 加 说 明 。 


5. 章节 


本 书 在 设计 时 认真 考虑 了 人 们 学 习 Java 语 言 的 方式 。 在 我 授课 时 ， 学 生 
们 的 反映 有 效 地 帮助 了 我 认识 哪些 部 分 是 比较 困难 的 ， 需 特别 加 以 留 
意 。 我 也 曾经 一 次 讲述 了 太 多 的 问题 ， 但 得 到 的 教训 是 : 假如 包括 了 大 
量 新 特性 ， 就 需要 对 它们 全 部 作出 解释 ， 而 这 特别 容易 加 深 学 生 们 的 混 
请。 因此 ， 我 进行 了 大 量 努 力 ， 使 这 本 书 一 次 尽 可 能 地 少 涉及 一 些 问 


el 


所 以 ， 我 在 书 中 的 目标 是 让 每 一 章 者 讲述 一 种 语言 特性 ， 或 者 只 讲述 少 
数 几 个 相互 天 联 的 特性 。 这 样 一 来 ， 读 者 在 转向 下 一 主题 时 ， 就 能 更 容 
易 地 消化 前 面 学 到 的 知识 。 


下 面 列 出 对 本 书 各 半 的 一 个 简要 说 明 ， 它 们 与 我 实际 进行 的 谍 堂 教学 古 
对 应 的 。 


(1) 第 1 章 : 对 象 入 门 


这 一 章 是 对 面向 对 象 的 程序 设计 (OOP) 的 一 个 综述 ， 其 中 包括 对 “ 什 
么 是 对 象 ” 之 类 的 基本 问题 的 回答 ， 并 讲述 了 接口 与 实现 、 抽 象 与 封 
六、 消 居 与 函数 、 继 承 与 合成 以 及 非常 重要 的 多 形 性 的 概念 。 这 一 章 会 
问 大 家 提出 一 些 对 象 创建 的 基本 问题 ， 比 如 构建 费 、 对 象 存 在 于 何 处 、 
创建 好 后 把 它们 置 于 什么 地 方 以 及 魔术 般 的 垃圾 收集 项 《〈 能 够 清除 不 再 
需要 的 对 象 ) 。 要 介绍 的 另 一 些 问 题 还 包括 通过 违例 实现 的 错误 控制 机 
制 、 反 应 灵敏 的 用 户 界 面 的 多 线程 处 理 以 及 连 网 和 因特网 等 等 。 大 家 也 
会 从 中 了 解 到 是 什么 使 得 Java 如 此 特别 ， 它 为 什么 取得 了 这 么 大 的 成 
功 ， 以 及 与 面向 对 象 的 分 析 与 设计 有 关 的 问题 。 


(2) 第 2 章 : 一 切 都 是 对 象 


本 章 将 大 家 人 带 到 可 以 着 手写 目 己 的 第 一 个 Java 程 序 的 地 方 ， 所 以 必须 对 
一 些 基 本 概念 作出 解释 ， 其 中 包括 对 象 “句柄 ”的 概念 ， 怎 样 创建 一 个 对 























象 ， 对 基本 数据 类 型 和 数组 的 一 个 介绍 ; 作用 域 以 及 垃圾 收集 器 清除 对 
象 的 方式 ， 如 何 将 Java 中 的 所 有 东西 都 归 为 一 种 新 数据 类 型 〈 类 ) ， 以 
及 如 何 创 建 自己 的 类 ; 函数 、 自 变量 以 及 返回 值 ， 名 字 的 可 见 度 以 及 使 
用 来 自 其 他 库 的 组 件 ，static 关 键 字 ; 注释 和 骸 入 文档 等 等 。 


(3) 第 3 章 : 控制 程序 流程 


本 章 开始 介绍 起 源 于 C 和 C++， 由 Java 继 承 的 所 有 运算 符 。 除 此 以 外 ， 
还 要 学 习 运 算 符 一 些 不 易 使 人 注意 的 问题 ， 以 及 涉及 造型 、 升 迁 以 及 优 
先 次 序 的 问题 。 随 后 要 讲述 的 是 基本 的 流程 控制 以 及 选择 运算 ， 这 些 是 
几乎 所 有 程序 设计 语言 都 具有 的 特性 : 用 if-else 实 现 选 择 ， 用 for 和 while 
实现 循环 ;用 break 和 continue 以 及 Java 的 标签 式 break 和 contiune (它们 被 
认为 是 Java 中 “不 见 的 gogo”) 退出 循环 ， 以 及 用 switch 实 现 男 一 种 形式 
的 选择 。 尺 管 这 些 与 C 和 C++ 中 见 到 的 有 一 定 的 共通 性 ， 但 多 少 存在 一 
些 区 别 。 除 此 以 外 ， 所 有 示例 都 是 完整 的 Java 示 例 ， 能 使 大 家 很 快 地 熟 
悉 Java 的 外 观 。 


(4) 第 4 章 : 初始 化 和 清除 


本 章 开 始 介绍 构建 器 ， 它 的 作用 是 担保 初始 化 的 正确 实现 。 对 构建 器 的 
定义 要 涉及 函数 过 载 的 概念 (因为 可 能 同时 有 几 个 构建 种) 。 随 后 要 讨 
论 的 是 清除 过 程 ， 它 并 非 肯 定 如 想象 的 那么 简单 。 用 完 一 个 对 象 后 ， 通 
常 可 以 不 必 管 它 ， 垃 圾 收集 器 会 目 动 介入 ， 释 放 由 它 占据 的 内 存 。 这 里 
详细 探讨 了 垃圾 收集 如 以 及 它 的 一 些 特点 。 在 这 一 章 的 最 后 ， 我 们 将 更 
贴近 地 观察 初始 化 过 程 : 目 动 成 员 初 始 化 、 指 定 成 员 初 始 化 、 初 始 化 的 
顺序 、static CHAS) 初始 化 以 及 数组 初始 化 等 等 。 


(5) 第 5 章 : 隐藏 实现 过 程 


本 章 要 探讨 将 代码 封装 到 一 起 的 方式 ， 以 及 在 库 的 其 他 部 分 隐藏 时 ， 为 
什么 仍 有 一 部 分 处 于 骏 露 状态 。 首 先 要 讨论 的 是 package 和 import 关 键 
字 ， 它 们 的 作用 是 进行 文件 级 的 封装 (打包 ) 操作 ， 并 允许 我 们 构建 由 
类 构成 的 库 〈 类 库 ) 。 此 时 也 会 谈 到 目录 路 径 和 文件 名 的 问题 。 本 章 剩 
下 的 部 分 将 讨论 public，private 以 及 protected 三 个 关键 字 、“ 友 好 ”访问 的 
概念 以 及 各 种 场合 下 不 同 访问 控制 级 的 意义 。 


(6) 第 6 章 : 类 再 生 




















继承 的 概念 是 几乎 所 有 OOP 语 言 中 都 占有 重要 的 地 位 。 它 是 对 现 有 类 加 
以 利用 ， 并 为 其 添加 新 功能 的 一 种 有 效 途 径 〈 同 时 可 以 修改 它 ， 这 是 第 
7 半 的 主题 》。 通 过 继承 来 重复 使 用 原 有 的 代码 时 (再生 )〉 ， 一 般 需 要 
保持 “基础 类 ”不 变 ， 只 是 将 这 儿 或 那儿 的 东西 串联 起 来 ， 以 达到 预期 的 
效果 。 然 而 ， 继 承 并 不 是 在 现 有 类 基础 上 制造 新 类 的 唯一 手段 。 通 

过 “合成 "”， 亦 可 将 一 个 对 象 局 入 新 类 。 在 这 一 半 中 ， 大 家 将 学 习 在 Java 
中 重复 使 用 代码 的 这 两 种 方法 ， 以 及 具体 如 何 运用 。 


(7) 第 7 章 : 多 形 性 


若 由 你 自己 来 干 ， 可 能 要 花 9 个 月 的 时 间 才 能 发 现 和 理解 多 形 性 的 问 
题 ， 这 一 特性 实际 是 OOP 一 个 重要 的 基础 。 通 过 一 些小 的 、 简 单 的 例 
子 ， 读 者 可 知道 如 何 通 过 继承 来 创建 一 系列 类 型 ， 并 通过 它们 共有 的 基 
础 类 对 那个 系列 中 的 对 象 进行 操作 。 通 过 Java 的 多 形 性 概念 ， 同 一 系列 
中 的 所 有 对 象 都 有 具有 了 共通 性 。 这 意味 着 我 们 编写 的 代码 不 必 再 依赖 特 
定 的 类 型 信息 。 这 使 程序 更 易 扩 展 ， 包 容 力 也 更 强 。 由 此 ， 程 序 的 构建 
和 代码 的 维护 可 以 变 得 更 方便 ， 付 出 的 代价 也 会 更 低 。 此 外 ，Java 还 通 
过 “接口 ”提供 了 设置 再 生 关系 的 第 三 种 途径 。 这 儿 所 谓 的 “接口 ?是 对 对 
象 物理 “接口 ”一 种 纯粹 的 抽象 。 一 旦 理解 了 多 形 性 的 概念 ， 接 口 的 含义 
束 很 容易 解释 了 。 本 章 也 同 大 家 介绍 了 Java 1.1 的 “内 部 类 ”。 


(8) 第 8 章 : 对 象 的 容纳 


对 一 个 非常 简单 的 程序 来 说 ， 它 可 能 只 拥有 一 个 固定 数量 的 对 象 ， 而 且 
对 象 的 “生存 时 间 ” 或 者 “存在 时 间 ” 是 已 知 的 。 但 是 通 第 ， 我 们 的 程序 会 
在 不 定 的 时 间 创 建新 对 象 ， 只 有 在 程序 运行 时 才 可 了 解 到 它们 的 详情 。 
此 外 ， 除 非 进入 运行 期 ， 否 则 无 法 知道 所 需 对 象 的 数量 ， 其 至 无 法 得 知 
它们 确切 的 类 型 。 为 解决 这 个 常见 的 程序 设计 问题 ， 我 们 需要 拥有 一 种 
能 力 ， 可 在 任何 时 间 、 任 何 地 点 创建 任何 数量 的 对 象 。 本 章 的 宗旨 便 是 
探讨 在 使 用 对 象 的 同时 用 来 容纳 它们 的 一 些 Java 工 具 : 从 简单 的 数组 到 
复杂 的 集合 (数据 结构 ) ， 如 Vector 和 Hashtable 等 。 最 后 ， 我 们 还 会 深 
入 讨论 新 型 和 改进 过 的 Java 1.2 集 合 库 。 


(9) BOR: 违例 差错 控制 
Java 最 基本 的 设计 守则 之 一 便 是 组 织 错误 的 代码 不 会 真 的 运行 起 来 。 编 


译 融 会 尺 可 能 捕获 问题 。 但 条 些 情况 下 ， 除 非 进 入 运行 期 ， 否 则 问题 古 
不 会 被 及 现 的。 这 些 问 题 要 么 属于 编程 错误 ， 要 么 则 是 一 些 目 然 的 出 错 
























































状况 ， 它 们 只 有 在 作为 程序 正常 运行 的 一 部 分 时 才 会 成 立 。Java 为 此 提 
供 了 “违例 控制 ”机制 ， 用 于 控制 程序 运行 时 产生 的 一 切 问题 。 这 一 章 将 
解释 try、catch、throw、throws 以 及 finally 等 关键 字 在 Java 中 的 工作 原 
理 。 并 讲述 什么 时 候 应 当 “ 掷 ?出 违例 ， 以 及 在 捕获 到 违例 后 该 采取 什么 
操作 。 此 外 ， 大 家 还 会 学 习 Java 的 一 些 标准 违例 ， 如 何 构建 自己 的 违 
例 ， 违 例 发 生 在 构建 器 中 怎么 人 办， 以 及 违例 控制 器 如 何 定位 等 等 。 


(10) 第 10 章 : Java IO 系统 


理论 上 ， 我 们 可 将 任何 程序 分 割 为 三 部 分 : 输入 、 处 理 和 输出 。 这 意味 
HIO 输入 / 输出 ) 是 所 有 程序 最 为 关键 的 部 分 。 在 这 一 章 中 ， 大 家 将 
学 习 Java 为 此 提供 的 各 种 类 ， 如 何 用 它们 读 写 文件 、 内 存 块 以 及 控制 台 
等 。“ 老 ”IO 和 Java 1.1 的 “新 "IO 将 得 到 着 重 强调 。 除 此 之 外 ， 本 节 还 要 探 
讨 如 何 获 取 一 个 对 象 、 对 其 进行 “ 流 式 ”加工 〈 使 其 能 置 入 破 盘 或 通过 网 
络 传送 ) 以 及 重新 构建 它 等 等 。 这 些 操作 在 Java 的 1.1 版 中 都 可 以 上 自动 完 
a 我 们 也 要 讨论 Java 1.1 的 压缩 库 ， 它 将 用 在 Java 的 归档 文件 格 
式 中 (JAR) 。 

















(11) 第 11 章 : 运行 期 类 型 鉴定 


若 只 有 指向 基础 类 的 一 个 句柄 ，Java 的 运行 期 类 型 标 鉴定 (RTT 使 我 
们 能 获知 一 个 对 象 的 准确 类 型 是 什么 。 一 般 情 况 下 ， 我 们 需要 有 意 忽略 
一 个 对 象 的 准确 类 型 ， 让 Java 的 动态 绑 定 机 制 〈 多 形 性 ) 为 那 一 类 型 实 
现 正 确 的 行为 。 但 在 茶 些 场合 下 ， 对 于 只 有 一 个 基础 句柄 的 对 象 ， 我 们 
仍然 特别 有 必要 了 解 它 的 准确 类 型 是 什么 。 拥 有 这 个 资料 后 ， 通 种 可 以 
更 有 效 地 执行 一 次 特殊 情况 下 的 操作 。 本 章 将 解释 RITI 的 用 途 、 如 何 
使 用 以 及 在 适当 的 时 候 如 何 放弃 它 。 此 外 ，Java 1.1 的 “反射 ?特性 也 会 在 
这 里 得 到 介绍 。 


(12) 第 12 章 : 传递 和 返回 对 象 


由 于 我 们 在 Java 中 同 对 象 沟通 的 唯一 途径 是 “句柄 ”， 所 以 将 对 象 传递 到 
一 个 函数 里 以 及 从 那个 函数 返回 一 个 对 象 的 概念 就 显得 非常 有 趣 了 。 本 
章 将 解释 在 函数 中 进出 时 ， 什 么 才 是 为 了 管理 对 象 需要 了 解 的 。 同 时 也 
会 讲述 String〈 字 串 ) 类 的 概念 ， 它 用 一 种 不 同 的 方式 解决 了 同样 的 问 
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(13) 第 13 章 : 创建 窗口 和 程序 片 


Java 配 套 提供 了 “抽象 Windows 工 具 包 ”(AWT) 。 这 实际 是 一 系列 类 的 
集合 ， 能 以 一 种 可 移植 的 形式 解决 视窗 操纵 问题 。 这 些 窗 口 化 程序 既 可 
以 程序 片 的 形式 出 现 ， 亦 可 作为 独立 的 应 用 程序 使 用 。 本 章 将 癌 大 家 介 
绍 AWT 以 及 网 上 程序 片 的 创建 过 程 。 我 们 也 会 探讨 AWTI 的 优 缺 点 以 及 
Java 1.1 在 GUI 方 面 的 一 些 改进 。 同 时 ， 重 要 的 “Java Beans” 技 术 也 会 在 
这 里 得 到 强调 。Java Beans 是 创建 “快速 应 用 开发 "(RAD) 程序 构造 工 
具 的 重要 基础 。 我 们 最 后 介绍 的 是 Java ”1.2 的 “Swing” 库 一 一 它 使 Java 的 
UI 组 件 得 到 了 显著 的 改善 。 


(14) 第 14 章 : 多 线程 


Java 提 供 了 一 套 内 建 的 机 制 ， 可 提供 对 多 个 并 有 友子 任务 的 支持 ， 我 们 称 
其 为 “线程 ”。 这 线程 均 在 单一 的 程序 内 运行 。 除 非 机 器 安装 了 多 个 处 理 
#， 人 奋 则 这 就 是 多 个 子 任务 的 唯一 运行 方式 。 尺 管 还 有 别 的 许多 重要 用 
途 ， 但 在 打算 创建 一 个 反应 灵敏 的 用 户 界 面 时 ， 多 线程 的 运用 显得 无 为 
重要 。 举 个 例子 来 说 ， 在 采用 了 多 线程 技术 后 ， 尽 管 当时 还 有 别 的 任务 
在 执行 ， 但 用 户 仍 然 可 以 这 无 阻碍 地 按 下 一 个 按钮 ， 或 者 键入 一 些 文 

字 。 本 章 将 对 Java 的 多 线程 处 理 机 制 进行 探讨 ， 并 介绍 相关 的 语法 。 


(15) 第 15 章 网 络 编程 


开始 编写 网 络 应 用 时 ， 就 会 发 现 所 有 Java 特 性 和 库 仿 佛 早已 串联 到 了 一 
起 。 本 章 将 探讨 如 何 通过 因特网 通信 ， 以 及 Java 用 以 辅助 此 类 编程 的 一 
些 类 。 此 外 ， 这 里 也 展示 了 如 何 创 建 一 个 Java 程 序 片 ， 令 其 同一 个 “ 通 
用 网 关 接 口 ”(CGI) 程序 通信 ; 揭示 了 如 何 用 C++ 编 写 CGI 程 序 ， 也 讲 
述 了 与 Java 1.1 的 “Java 数 据 库 连 接 *(JDBC) 和 “远程 方法 调用 ”(RMI) 
有 关 的 问题 。 


(16) 第 16 章 设计 范式 

本 章 将 讨论 非常 重要 、 但 同时 也 是 非 传统 的 “范式 ”程序 设计 概念 。 大 家 
会 学 习 设计 进展 过 程 的 一 个 例子 。 首 先是 最 初 的 方案 ， 然 后 经 历 各 种 程 
序 逻 辑 ， 将 方案 不 断 改革 为 更 恰当 的 设计 。 通 过 整个 过 程 的 学 习 ， 大 家 
可 体会 到 使 设计 思想 逐渐 变 得 清晰 起 来 的 一 种 途径 。 

(17) 第 17 章 m H 


本 草包 括 了 一 系列 项 目 ， 它 们 要 么 以 本 书 前 面 讲述 的 内 容 为 基础 ， 要 人 么 


























对 以 前 各 章 进行 了 一 番 扩 展 。 这 些 项 目 显 然 是 书 中 最 复杂 的 ， 它 们 有 效 
演示 了 新 技术 和 类 库 的 应 用 。 


有 些 主题 似乎 不 太 适 合 放 到 本 书 的 核心 位 置 ， 但 我 友 现 有 必要 在 教学 时 
讨论 它们 ， 这 些 主题 部 放 入 了 本 书 的 附录 。 


(18) 附录 A: 使 用 非 Java 代 码 


对 一 个 完全 能 够 移植 的 Java 程 序 ， 它 肯定 存在 一 些 严重 的 缺陷 ， 速 度 太 
慢 ， 而 且 不 能 访问 与 具体 平台 有 关 的 服务 。 若 事先 知道 程序 要 在 什么 平 
台 上 使 用 ， 就 可 考虑 将 一 些 操 作 变 成 * 回 有 方法 ”， 从 而 显著 加 快 执行 速 
度 。 这 些 “ 固 有 方法 ”实际 是 一 些 特殊 的 函数 ， 以 另 一 种 程序 设计 语言 写 
成 〈 目 前 仅 支 持 C/C++) 。Java 还 可 通过 另 一 些 途 径 提 供 对 非 Java 代 码 
的 支持 ， 其 中 包括 CORBA。 本 附录 将 详细 介绍 这 些 特性 ， 以 便 大 家 能 
创建 一 些 简单 的 例子 ， 同 非 Java 代 码 打交道 。 


(19) 附录 B: 对 比 C++ 和 Java 

对 一 个 C++ 程序 员 ， 他 应 该 已 经 掌握 了 面向 对 象 程 序 设 计 的 基本 概念 ， 
而 且 Java 语 法 对 他 来 说 无 疑 是 非常 眼熟 的 。 这 一 点 是 明显 的 ， 因 为 Java 
本 身 就 是 从 C++ 衍生 而 来 。 但 是 ，C++ 和 Java 之 间 的 确 存 在 一 些 显著 的 
差异 。 这 些 差 异 意 味 着 Java 在 C++ 基础 上 作出 的 重大 改进 。 一 旦 理解 了 
这 些 差 异 ， 就 能 理解 为 什么 说 Java 是 一 种 杰出 的 语言 。 这 一 附录 便 是 为 
这 个 目的 设立 的 ， 它 讲述 了 使 Java 与 C++ 明显 有 别 的 一 些 重 要 特性 。 
(20) 附录 C: Java 编 程 规则 

本 附录 提供 了 大 量 建议 ， 帮 助 大 家 进行 低级 程序 设计 和 代码 编写 。 

(21) 附录 D: 性 能 


通过 这 个 附录 的 学 习 ， 大 家 可 发 现 目 己 Java 程 序 中 存在 的 瓶颈 ， 并 可 有 
效 地 改善 执行 速度 。 


(22) 附录 E: 关于 垃圾 收集 的 一 些 话 
这 个 附录 讲述 了 用 于 实现 垃圾 收集 的 操作 和 方法 。 
(23) 附录 F: 推荐 读物 






































列 出 我 感觉 特别 有 用 的 一 系列 Java 参 考 书 。 
6. 练习 


为 她 固 对 新 知识 的 和 营 握 ， 我 发 现 简单 的 练习 特别 有 用 。 所 以 读者 在 每 一 
章 结束 时 都 能 找到 一 系列 练习 。 


大 多 数 练习 都 很 简单 ， 在 合理 的 时 间 内 可 以 完成 。 如 将 本 书 作为 教材 ， 
可 考虑 在 谍 符 内 完成 。 老 师 要 注意 观察 ， 确 定 所 有 学 生 都 已 消化 了 讲授 
的 内 容 。 有 些 练习 要 难 些 ， 他 们 是 为 那些 有 兴趣 深入 的 读者 准备 的 。 大 
多 数 练习 都 可 在 较 短 时 间 内 做 完 ， 有 效 地 检测 和 加 深 您 的 知识 。 有 些 题 
目 比 较 具 有 挑战 性 ， 但 都 不 会 太 麻 烦 。 事 实 上 ， 练 习 中 辜 到 的 问题 在 实 
际 应 用 中 也 会 经 常 碰 到 。 


7. 多 媒体 CD-ROM 

本 书 配套 提供 了 一 片 多 媒体 CD-ROM， 可 单独 购买 及 使 用 。 它 与 其 他 计 
算 机 书籍 的 普通 配套 CD 不 同 ， 那 些 CD 通 常 仅 包含 了 书 中 用 到 的 源码 
(本 书 的 源码 可 从 www.BruceEckel.com 免 费 下 载 ) 。 本 CD-ROM 是 一 个 
独立 的 产品 ， 包 含 了 一 周 “Hads-OnjJava” 培 训 课 程 的 全 部 内 容 。 这 是 一 
个 由 Bruce ” Eckel 讲授 的 、 长 度 在 15 小 时 以 上 的 课程 ， 含 500 张 以 上 的 演 
PRATH F o 该 课程 建立 在 这 本 书 的 基础 上 ， 所 以 是 非常 理想 的 一 个 配套 
产品 。 

CD-ROM 包 含 了 本 书 的 两 个 版 本 : 

(1) 本 书 一 个 可 打印 的 版 本 ， 与 下 载 版 完全 一 致 。 


(2) “为 方便 读者 在 屏幕 上 阅读 和 索引 ，CD-ROM 提 供 了 一 个 独特 的 超 链 
接 版 本 。 这 些 超 链接 包括 : 


加 230 个 半 、 闻 和 小 标题 链接 
四 3600 个 索引 链接 


CD-ROM 刻 录 了 600MB 以 上 的 数据 。 我 相信 它 已 对 所 谓 “ 物 超 所 值 ”进行 
了 般 新 的 定义 。 


CD-ROM 人 包含 了 本 书 打印 版 的 所 有 东西 ， 为 外 还 有 来 日 五 天 快速 入 门 诬 





程 的 全 部 材料 。 我 相信 它 建立 了 一 个 新 的 书刊 品质 评定 标准 。 


若 想 单独 购买 此 CD-ROM， 只 能 从 Web 站 点 www.BruceEckel.com 处 直接 
订购 。 


8. 源 代码 


本 书 所 有 源码 都 作为 保留 版 权 的 免费 软件 提供 ， 可 以 独立 软件 包 的 形式 
获得 ， 亦 可 从 http://www.BruceEckel.com 下 载 。 为 保证 大 家 获得 的 是 最 
新 版 本 ， 我 用 这 个 正式 站 点 发 行 代码 以 及 本 书 电子 版 。 亦 可 在 其 他 站 点 
找到 电子 书 和 源码 的 镜像 版 (有些 站 点 已 在 http://www.BruceEckel.com 
处 列 出 ) 。 但 无 论 如 何 ， 都 应 检查 正式 站 点 ， 确 定 镜像 版 确实 是 最 新 的 
版 本 。 可 在 课堂 和 其 他 教育 场所 发 布 这 些 代码 。 


版 权 的 主要 目标 是 保证 源码 得 到 正确 的 引用 ， 并 防止 在 未 经 许可 的 情况 
下 ， 在 印刷 材料 中 发 布 代码 。 通 向， 只 要 源码 获得 了 正确 的 引用 ， 则 在 
大 多 数 媒体 中 使 用 本 书 的 示例 都 没有 什么 问题 。 


在 每 个 源码 文件 中 ， 都 能 发 现下 述 版 本 声明 文字 : 
16-17 页 程序 


可 在 自己 的 开发 项 目 中 使 用 代码 ， 并 可 在 读 莹 上 引用 《包括 学 习 材 
料 ) 。 但 要 确定 版 权 声明 在 每 个 源 文件 中 得 到 了 保留 。 


9. 编码 样式 


在 本 书 正 文中 ， 标 识 符 〈 函 数 、 变 量 和 类 名 ) 以 粗 体 印刷 。 大 多 数 关 键 
字 也 采用 粗 体 一 一 除了 一 些 频 楷 用 到 的 关键 字 〈 大 全 部 采用 粗 体 ， 会 使 
页 面 拥挤 难看 ， 比 如 那些 “类 ”) 。 


对 于 本 书 的 示例 ， 我 采用 了 一 种 特定 的 编码 样式 。 该 样式 得 到 了 大 多 数 
Java 开 发 环境 的 文 持 。 该 样式 问世 己 有 几 年 的 时 间 ， 最 早起 源 于 Bjarne 
Stroustrup 先 生 在 《The C++ Programming Language》 里 采用 的 样式 
(Addison-Wesley 1991 年 出 版 ， 第 2 版 ) 。 由 于 代码 样式 目前 是 个 敏感 
问题 ， 极 易 招 致 数 小 时 的 激烈 辩论 ， 所 以 我 在 这 儿 只 想 指 出 自己 并 不 打 
算 通 过 这 些 示 例 建立 一 种 样式 标准 。 之 所 以 采用 这 些 样式 ， 完 全 出 于 我 
自己 的 考虑 。 由 于 Java 是 一 种 形式 非常 自由 的 编程 语言 ， 所 以 读者 完全 























可 以 根据 上 自己 的 感觉 选用 了 适合 的 编码 样式 。 


本 书 的 程序 是 由 字 人 处 理 程 序 包 括 在 正文 中 的 ， 它 们 直接 取 目 编译 好 的 文 
件 。 所 以 ， 本 书 印 刷 的 代码 文件 应 能 正常 工作 ， 不 会 造成 编译 器 错误 。 
会 造成 编译 错误 的 代码 已 经 用 注释 //! 标 出 。 所 以 很 容易 发 现 ， 也 很 容易 
用 自动 方式 进行 测试 。 读 者 发 现 并 回 作 者 报告 的 错误 首先 会 在 发 行 的 源 
人 码 中 改正 ， 然 后 在 本 书 的 更 新 版 中 校订 (所 有 更 新 都 会 在 Web 站 点 
http://www.BruceEckel.com 处 出 现 ) 。 


10. Java 版 本 


尽管 我 用 几 家 厂商 的 Java 开 发 平台 对 本 书 的 代码 进行 了 测试 ， 但 在 判断 
代码 行为 是 否 正确 时 ， 却 通常 以 Sun 公 司 的 Java 开 发 平台 为 准 。 


当 您 读 到 本 书 时 ，Sun 应 已 发 行 了 Java 的 三 个 重要 版 本 : 1.0, 1.1% 

1.2 Sun 声称 每 9 个 月 就 会 发 布 一 个 主要 更 新 版 本 ) 。 就 我 看 ，1.1 厂 对 
Java 语 言 进行 了 显 着 改进 ， 完 全 应 标记 成 2.0 版 “由 于 1.1 已 作出 了 如 此 
大 的 修改 ， 真 不 敢 想 象 2.0 版 会 出 现 什么 变化 )。 然 而 ， 它 的 1.2 版 看 起 
来 最 终 将 Java 推 入 了 一 个 全 盛 时 期 ， 特 别 是 其 中 考 夸 到 了 用 户 界 面 工 
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本 书 主要 讨论 了 1.0 和 1.1 版 ，1.2 版 有 部 分 内 容 涉及 。 但 在 有 些 时 候 ， 新 
方法 明显 优 于 老 方法 。 此 时 ， 我 会 明显 偏 问 于 新 方法 ， 通 常 教 给 大 家 更 
好 的 方法 ， 而 完全 忽略 老 方 法 。 然 而 ， 有 的 新 方法 要 以 老 方 法 为 基础 ， 
所 以 不 可 避免 地 要 从 老 方法 入 手 。 这 一 特点 尤 以 AWT 为 其 ， 因 为 那儿 
不 仪 存在 数量 众多 的 老式 Java 1.0 代码， 有 的 平台 仍然 只 文 持 Java 1.0. 
我 会 尽量 指出 哪些 特性 是 哪个 版 本 特有 的 。 


大 家 会 注意 到 我 并 未 使 用 子 版 本 号 ， 比 如 1.1.1。 至 本 书 完稿 为 止 ，Sun 
公司 发 布 的 最 后 一 个 1.0 版 是 1.02， 而 1.1 的 最 后 版 本 是 1.1.5 (Java 1.2 仍 
在 做 测试 ) 。 在 这 本 书 中 ， 我 只 会 提 到 Java 1.0, Java 1.1 及 Java 1.2, 
避免 由 于 子 版 本 编号 过 多 造成 的 键入 和 印刷 错误 。 


11. 课程 和 培训 
我 的 公司 提供 了 一 个 五 日 制 的 公共 培训 课程 ， 以 本 书 的 内 容 为 基础 。 每 


章 的 内 容 都 代表 着 一 尝 读 ， 并 附 有 相应 的 谍 后 练习 ， 以 便 苑 固 学 到 的 知 
识 。 一 些 辅助 用 的 约 灯 片 可 在 本 书 的 配套 光盘 上 找到 ， 最 大 限度 地 方便 





























各 位 读者 。 欲 了 解 更 多 的 情况 ， 请 访问 : 
http:/www.BruceEckel.com 

或 友 函 至 

Bruce@EckelObjects.com 


我 的 公司 也 提供 了 咨询 服务 ， 指 导 客 户 完成 整个 开发 过 程 一 一 特别 是 您 
的 单位 首次 接触 Java 开 发 的 时 候 。 


12. 错误 


At 人 作者 花 多 大 精力 来 避免 ， 错 误 总 是 从 意 想不到 的 地 方 冒 出 来 。 如 果 
您 认为 自己 发 现 了 一 个 错误 ， 请 在 源 文件 (可 在 

the// www Brucekckelicom tiik F] ) 里 指出 有 可 能 是 错误 的 地 方 ， 填 好 
我 们 提供 的 表单 。 将 您 推荐 的 纠 错 方 法 通过 电子 函件 发 给 

Bruce@EckelObjects.com。 经 适当 的 核对 与 处 理 ，Web 站 点 的 电子 版 以 
及 本 书 的 下 一 个 印刷 版 本 会 作出 相应 的 改正 。 具 体格 式 如 下 : 


(1) 在 主题 行 《Subject) 写 上 “TIJ Correction”( 去 掉 引 号 ) ， 以 便 您 的 函 
件 进 入 对 应 的 目录 。 


(2) 在 函件 正文 ， 采用 下 述 形式 : 
find: 在 这 里 写 一 个 单行 字 串 ， 以 便 我 们 搜索 错误 所 在 的 地 方 


Comment: 在 这 里 可 写 多 行 批注 正文 ， 最 好 以 “here's how I think it shoud 
read” FF $ 


HHH 


其 中 ，“###” 指 出 批注 正文 的 结束 。 这 样 一 来 ， 我 自己 设计 的 一 个 纠 错 
工具 就 能 对 原始 正文 来 一 次 “搜索 "， 而 您 建议 的 纠 错 方法 会 在 随后 的 一 
个 窗口 中 弹出 。 


奉 希 望 在 本 书 的 下 一 版 添加 什么 内 容 ， 或 对 书 中 的 练习 题 有 什么 意见 
也 欢迎 您 指出 。 我 们 感谢 您 的 所 有 意见 。 











13. 封面 设计 


«Thinking in Java》 一 书 封面 的 创作 灵感 来 源 于 American Arts & 
CraftsMovement 〈 美 洲 艺 术 区 手工 艺 品 运动 ) 。 这 一 运动 起 始 于 世纪 之 
交 ，1900 到 1920 年 达到 了 顶峰 。 它 起 源 于 英格兰 ， 具 有 一 定 的 历史 背 
景 。 当 时 正 是 机 器 革命 产生 的 风暴 席卷 整个 大 陆 的 时 候 ， 而 且 受 到 维 多 
利 亚 地 区 强烈 装饰 风格 的 巨大 影响 。Arts&Crafts 强 调 的 是 原始 风格 ， 回 
归 上 自然 的 初衷 是 整个 运动 的 核心 。 那 时 对 手工 制作 推 法 备至 ， 手 工艺 人 
特别 得 到 尊重 。 正 因为 如 此 ， 人 们 远 远 避 开 现代 工具 的 使 用 。 这 场 运 动 
对 整个 艺术 界 造成 了 深远 的 影响 ， 直 至 今天 仍 受 到 人 们 的 怀念 。 特 别 是 
我 们 面临 又 一 次 世纪 之 交 ， 强 烈 的 怀旧 情绪 难免 涌 上 心 来 。 计 算 机 发 展 
至 今 ， 已 走 过 了 很 长 的 一 段 路 。 我 们 更 迫切 地 感到 : 软件 设计 中 最 重要 
的 是 设计 者 本 有 身 ， 而 不 是 流水 化 的 代码 编制 。 如 设计 者 本 喘 的 素质 和 修 
养 不 高 ， 那 么 最 多 只 是 “生产 ”代码 的 工具 而 已 。 


我 以 同样 的 眼光 来 看 得 Java: 作为 一 种 将 程序 员 从 操作 系统 繁琐 机 制 中 
解放 出 来 的 答 试 ， 它 的 目的 是 使 人 们 成 为 真正 的 “软件 艺术 家 ?”。 


无 论 作 者 还 是 本 书 的 封面 设计 者 《〈 目 孩提 时 代 就 是 我 的 朋友 ) 都 从 这 一 
场 运动 中 获得 了 灵感 。 所 以 接 下 来 的 事情 就 非常 简单 了 ， 要 么 目 己 设 
计 ， 要 么 直接 采用 来 自 那 个 时 期 的 作品 。 


此 外 ， 封 面 癌 大 家 展示 了 一 个 收集 箱 ， 目 然 学 者 可 能 用 它 展 示 目 己 的 昆 
虫 标本 。 我 们 认为 这 些 昆虫 都 是 "对象 "， 全 部 置 于 更 大 的 “收集 箱 ? 对 象 
里 ， 再 统一 置 入 “封面 > 这 个 对 象 里 。 它 问 我 们 揭示 了 面 癌 对 象 编程 扩 术 
最 基本 的 “集合 概念。 当然 ， 作 为 一 名 程序 员 ， 大 家 对 于 “ 昆 

虫 ”或 “ 虫 ” 是 非常 敏感 的 (“ 虫 " 在 英语 里 是 Bug， 后 指 程序 错误 ) 。 这 里 
的 “ 虫 ” 已 被 抓获 ， 在 一 只 广 口 瓶 中 杀 死 ， 最 后 禁闭 于 一 个 小 的 展览 盒 

首 示 Java 有 能 力 寻 找 、 显 示 和 消除 程序 里 的 “ 虫 ”( 这 是 Java 最 上 其 特 
色 的 特性 之 一 ) 。 


14. 致谢 


首先 ， 感 谢 Doyle Street Cohousing Community 〈 道 尔 街 住 房 社区 ) 容忍 
我 花 两 年 的 时 间 来 写 这 本 书 ( 其 实 他 们 一 直 都 在 容 妨 我 的 “ 胡 做 非 

A”) 。 非 常 感谢 Kevin 和 Sonda Donovan， 是 他 们 把 科罗拉多 Crested 
Butte 市 这 个 风景 优美 的 地 方 租 给 我 ， 使 我 整个 夏天 都 能 安心 写作 。 感 谢 
Crested Butte 友 好 的 居民 ; 以 及 Rocky Mountain Biological 












































Laboratory〈 兰 石山 生物 实验 室 ) ， 他 们 的 工作 人 员 总 是 面 带 微笑 。 


这 是 我 第 一 次 找 代 理 人 出 书 ， 但 却 绝 没有 后 悔 。 谢 谢 * 摩 尔 文学 代理 公 
司 ” 的 Claudette Moore 小 姐 。 是 她 强大 的 信心 与 角力 使 我 最 终 梦 想 成 真 。 


我 的 头 两 本 书 是 与 Osborne/McGraw-Hi 刘 出 版 社 的 编辑 Jeff Pepper 合作 出 
版 的 。Jeff 又 在 正确 的 地 方 和 正确 的 时 间 出 现在 了 Prentice-Hall 出 版 社 ， 
是 他 为 了 清除 了 所 有 可 能 遇 到 的 障碍 ， 也 使 我 感受 了 一 次 愉快 的 出 书 经 
历 。 谢 谢 你 ，Jeff 一 你 对 我 非常 重要 。 


要 特别 感谢 Gen Kiyooka 和 他 的 Digigami 公 司 ， 我 用 的 Web 服 务 器 就 是 他 
们 提供 的 ; 也 要 感谢 Scott Callaway， 服 务 器 是 由 他 负责 维护 的 。 在 我 学 
习 Web 的 过 程 中 ， 一 个 服务 嚣 无疑 是 相当 有 价值 的 帮助 。 


谢谢 Cay Horstmann (《Core Java》 一 书 的 副 编辑 ，Prentice Hall 于 1997 
年 出 版 ) D'Arcy Smith (Symantec 公司 ) 和 Paul Tyma ( (Java Primer 
Plus》 一 书 的 副 编 辑 ，The Waite Group 于 1996 年 出 版 ) ， 感 谢 他 们 帮助 
我 澄清 语言 方面 的 一 些 概 念 。 


感谢 那些 在 “Java 软 件 开 发 会 议 ” 上 我 的 Java 小 组 发 言 的 同志 们 ， 以 及 我 
教授 过 的 那些 学 生 ， 他 们 提出 的 问题 使 我 的 教案 人 印发 成 熟 起 来 。 


特别 感谢 Larry 和 Tina O'Brien， 是 他 们 将 这 本 书 和 我 的 教学 内 容 制 成 一 
张 教学 CD-ROM (关于 这 方面 的 问题 ，http://www.BruceEckel.com 有 更 
多 的 答案 ) 。 


有 许多 人 送 来 了 纠 错 报告 ， 我 真 的 很 感激 所 有 这 些 朋 友 ， 但 特别 要 对 下 
面 这 些 人 说 声 谢谢 : Kevin Raulerson (发 现 了 多 处 重大 错误 ) ，Bob 
Resendes (发 现 的 错误 令 人 难以 置信 ) ，John Pinto, Joe Dante, Joe 
Sharp, David ”Combs〔( 许 多 语法 和 表达 不 清 的 地 方 ) > Dr. Robert 
Stephenson, Franklin Chen, Zev Griner, David Karr, Leander A. 
Stroschein, Steve Clark, Charles A. Lee, AustinMaher, Dennis P. 
Roth, Roque Oliveira, Douglas Dunn, Dejan Ristic, NeilGalarneau, 
David B. Malkovsky, Steve Wilkinson， 以 及 其 他 许多 热心 读者 。 


为 了 使 这 本 书 在 欧洲 发 行 ，Prof. Ir. Marc Meurrens 进 行 了 大 量 工 作 。 
有 一 些 技术 人 员 曾 走 进 我 的 生活 ， 他 们 后 来 都 和 我 成 了 朋友 。 最 不 寻常 























的 是 他 们 全 是 素食 主义 者 ， 平 时 喜欢 练习 瑜珈 功 ， 以 及 另 一 些 形 式 的 精 
神 训练 。 我 在 练习 了 以 后 ， 沉 得 对 我 保持 精力 的 旺盛 非常 有 好 处 。 他 们 
是 Kraig Brockschmidt，GenKiyooka 和 Andrea provaglio， 是 这 些 朋 友 帮 
我 了 解 了 Java 和 程序 设计 在 意大利 的 情况 。 


显然 ， 在 Delphi 上 的 一 些 经 验 使 我 更 容易 理解 Java， 因 为 它们 有 许多 概 
念 和 语言 设计 决定 是 相通 的 。 我 的 Delphi 朋 友 提 供 了 许多 帮助 ， 使 我 能 
够 洞察 一 些 不 易 为 人 注意 的 编程 环境 。 他 们 是 Marco ”Cantu〈( 男 一 个 意 
大 利 人 一 一 难道 会 说 拉丁 语 的 人 在 学 习 Java 时 有 得 天 独 厚 的 优势 ? ) 、 

Neil Rubenking (他 最 喜欢 瑜 匣 / 素食 / 禅 道 ， 但 也 非常 喜欢 计算 机 ) 
以 及 Zack Urlocker( 是 我 游历 世界 时 碰面 次 数 最 多 的 一 位 同志 ) 。 


我 的 朋友 Richard Hale Shaw〔 以 及 Kim) 的 一 些 意 见 和 支持 发 挥 了 非常 
关键 的 作用 。Richard 和 我 花 了 数 月 的 时 间 将 教学 内 容 合 并 到 一 起 ， 并 探 
讨 如 何 使 学 生 感 受到 最 完美 的 学 习 体 验 。 也 要 感谢 KoAnn Vikoren, Eric 
Eaurot, DeborahSommers, Julie Shaw, Nicole Freeman, Cindy Blair, 

Barbara Hanscome, Regina Ridley, Alex Dunne 以 及 MFI 其 他 可 冤 的 成 
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书籍 设计 、 封 面 设计 以 及 封面 照片 是 由 我 的 朋友 Daniel Will-Harris 制 作 
的 。 他 是 一 位 著名 的 作家 和 设计 家 Chttp://www.WillHarris.com) ， 在 初 
中 的 时 候 束 已 显露 出 了 过 人 的 数学 天 赋 。 但 是 ， 小 样 是 由 我 制作 的 ， 所 
以 录入 错误 都 是 我 的 。 我 是 用 Microsoft Word 97 for Windows 来 写 这 本 
书 ， 并 用 它 生 成 小 样 。 正 文字 体 采 用 的 是 Bitstream Carmina; 标题 采用 
Bitstream Calligraph 421 (www.bitstream.com) ; 每 章 开 头 的 符号 采用 
的 是 来 自 P22 的 Leonardo Extras (http:/www.p22.com) ; 封面 字体 采用 
IIC Rennie Marckintosh 。 


感谢 为 我 提供 编译 器 程序 的 一 些 闭 名 公司 : Borland, Microsoft, 
Symantec，Sybase/Powersoft/Watcom 以 及 Sun。 


特别 感谢 我 的 老师 和 我 所 有 的 学 生 《 他 们 也 是 我 的 老师 ) ， 其 中 最 有 趣 
的 一 位 写作 老师 是 Gabrielle Rico ( «(Writing the Natural Way》 一 书 的 作 
者 ，Putnam 于 1983 年 出 版 ) 。 


问 我 提供 过 文 持 的 朋友 包括 〈 当 然 还 不 止 ) : Andrew Binstock， 
SteveSinofsky, JD Hildebrandt, Tom Keffer, Brian McElhinney, 
Brinkley Barr, (Midnight Engineering) 278a¢tHYBill Gates, Larry 


























Constantine 和 LucyLockwood，Greg Perry, Dan Putterman, Christi 
Westphal， Gene Wang，DaveMayer，David Intersimone, Andrea 
Rosenfield, Claire Sawyers， 男 一 些 意大利 朋友 (Laura Fallai, 
Corrado, llsa 和 Cristina Giustozzi) ，Chris 和 Laura Strand, Almquists, 
Brad Jerbic, Marilyng Cvitanic, Mabrys, Haflingers, Pollocks, Peter 
Vinci, Robbins Families, Moelter Families (#UMcMillans) , Michael 
Wilk, Dave Stoner, Laurie Adams, Cranstons, Larry Fogg, Mike 和 
Karen Sequeira, Gary Entsmingerfl Allison Brody, KevinDonovan#ll 
Sonda Eastlack，Chester 和 Shannon Andersen, Joe Lordi, Dave 和 Brenda 
Bartlett, David Lee, Rentschlers, Sudeks, Dick, Patty 和 和 Lee Eckel, 
Lynn 和 Todd 以 及 他 们 的 家 人 。 最 后 ， 当 然 还 有 我 的 爸爸 和 妈妈 。 


Ate q 3. y 
第 1 音 对 象 入 门 
“为 什么 面向 对 象 的 编程 会 在 软件 开发 领域 造成 如 此 震 憾 的 影响 ? ” 


面 问 对 象 编程 《OOP) 具有 多 方面 的 吸引 力 。 对 管理 人 员 ， 它 实现 了 更 
快 和 更 廉价 的 开发 与 维护 过 程 。 对 分 析 与 设计 人 员 ， 建 模 处 理 变 得 更 加 
简单 ， 能 生成 清晰 、 易 于 维护 的 设计 方案 。 对 程序 员 ， 对 象 模型 显得 如 
此 高 雅 和 浅显 。 此 外 ， 面 癌 对 象 工具 以 及 库 的 巨大 威力 使 编程 成 为 一 项 
更 使 人 恰 悦 的 任务 。 每 个 人 都 可 从 中 获 益 ， 至 少 表面 如 此 。 


如 琳 说 它 有 缺点 ， 那 束 古 千 握 它 害 付出 的 代价 。 上 思考 对 象 的 时 候 ， 需 要 

采用 形象 思维 ， 而 不 是 程序 化 的 思维 。 与 程序 化 设计 相 比 ， 对 象 的 设计 

过 程 更 具 挑 战 性 一 一 特别 是 在 笃 试 创建 可 重复 使 用 (可 再 生 〉 的 对 象 

a E E 
F: 


(1) 选择 一 种 诸如 Smalltalk 的 语言 , “出 师 ? 前 必须 和 掌握 一 个 巨型 的 库 。 


(2) ”选择 几乎 根本 没有 库 的 C++ GERD) ， 然 后 深入 学 习 这 种 语言 
直至 能 自行 编写 对 象 库 。 


O: 幸运 的 是 ， 这 一 情况 已 有 明显 改观 。 现 在 有 第 三 方 库 以 及 标准 的 
C++ 库 供 选用 。 


事实 上 ， 很 难 很 好 地 设计 出 对 象 一 一 从 而 很 难 设计 好 任何 东西 。 因 此 ， 
只 有 数量 相当 少 的 “专家 ”能 设计 出 最 好 的 对 象 ， 然 后 让 其 他 人 享用 。 对 
于 成 功 的 OOP 语 言 ， 它 们 不 仅 集 成 了 这 种 语言 的 语法 以 及 一 个 编译 程序 
(iat) ， 而 且 还 有 一 个 成 功 的 开发 环境 ， 其 中 包含 设计 优 民 、 易 于 
使 用 的 库 。 所 以 ， 大 多 数 程序 员 的 首要 任务 就 是 用 现 有 的 对 象 解决 目 己 
的 应 用 问题 。 本 章 的 目标 就 是 癌 大 家 揭示 出 面 辣 对 象 编程 的 概念 ， 并 证 
明 它 有 多 么 简单 。 

本 章 将 回 大 家 解释 Java 的 多 项 设计 思想 ， 并 从 概念 上 解释 面 同 对 象 的 程 
序 设计 。 但 要 注意 在 阅读 完 本 章 后 ， 并 不 能 立即 编写 出 全 功能 的 Java 程 
序 。 所 有 详细 的 说 明和 示例 会 在 本 书 的 其 他 间 市 慢 慢 道 来 。 
























































1.1 抽象 的 进步 


所 有 编程 语言 的 最 终 目 的 都 是 提供 一 种 “抽象 方法。 一 种 较 有 和 争议 的 说 
法 是 : 解决 问题 的 复杂 程度 直接 取决 于 抽象 的 种 类 及 质量 。 这 儿 的 “种 
类 ”是 指 准备 对 什么 进行 “抽象 ?? 汇 编 语言 是 对 基础 机 器 的 少量 抽象 。 

后 来 的 许多 “命令 式 ” 语 言 (如 FORTRAN，BASIC 和 C) 是 对 汇编 语言 的 
一 种 抽象 。 与 汇编 语言 相 比 ， 这 些 语言 已 有 了 长 足 的 进步 ， 但 它们 的 抽 
象 原理 依然 要 求 我 们 着 重 考虑 计算 机 的 结构 ， 而 非 考 虑 问题 本 身 的 结 

构 。 在 机 器 模型 (位 于 “方案 空间 ”) 与 实际 解决 的 问题 模型 (位 于 “ 问 

题 空间 ”) 之 间 ， 程 序 员 必须 建立 起 一 种 联系 。 这 个 过 程 要 求人 们 付出 
较 大 的 精力 ， 而 且 由 于 它 脱离 了 编程 语言 本 身 的 范围 ， 造 成 程序 代码 很 
难 编写 ， 而 且 要 花 较 大 的 代价 进行 维护 。 由 此 造成 的 副作用 便 是 一 门 完 
善 的 “编程 方法 "学科 。 


为 机 器 建 模 的 男 一 个 方法 是 为 要 解决 的 问题 制作 模型 。 对 一 些 早期 语言 
来 说 ， 如 LISP 和 APL， 它 们 的 做 法 是 “从 不 同 的 角度 观察 世界 ”一 “所 

有 问题 都 归纳 为 列表 ”或 < 所 有 问题 都 归纳 为 算法 >”。PROLOG 则 将 所 有 
问题 都 归纳 为 决策 链 。 对 于 这 些 语言 ， 我 们 认为 它们 一 部 分 是 面向 基 

于 “强制 ”的 编程 ， 另 一 部 分 则 是 专 为 处 理 图 形 符号 设计 的 。 每 种 方法 都 
有 自己 特殊 的 用 途 ， 适 合 解决 某 一 类 的 问题 。 但 只 要 超出 了 它们 力 所 能 
及 的 范围 ， 就 会 显得 非常 笨拙 。 


面 问 对 象 的 程序 设计 在 此 基础 上 则 跨 出 了 一 大 步 ， 程 序 员 可 利用 一 些 工 
其 表达 问题 空间 内 的 元 素 。 由 于 这 种 表达 非常 普遍 ， 所 以 不 必 受 限于 特 
定 类 型 的 问题 。 我 们 将 问题 空间 中 的 元 素 以 及 它们 在 方案 空间 的 表示 物 
PEITZ” (Object) 。 当 然 ， 还 有 一 些 在 问题 空间 没有 对 应 体 的 其 他 
对 象 。 通 过 添加 新 的 对 象 类 型 ， 程 序 可 进行 灵活 的 调整 ， 以 便 与 特定 的 
问题 配合 。 所 以 在 阅读 方案 的 描述 代码 时 ， 会 读 到 对 问题 进行 表达 的 话 
语 。 与 我 们 以 前 见 过 的 相 比 ， 这 无 疑 是 一 种 更 加 灵活 、 更 加 强大 的 语言 
抽象 方法 。 总 之 ，OOP 允 许 我 们 根据 问题 来 描述 问题 ， 而 不 是 根据 方 

案 。 然 而 ， 仍 有 一 个 联系 途径 回 到 计算 机 。 每 个 对 象 都 类 似 一 台 小 计算 
机 ; 它们 有 自己 的 状态 ， 而 且 可 要 求 它们 进行 特定 的 操作 。 与 现实 世界 
的 “对 象 ” 或 者 “物体 * 相 比 ， 编 程 “ 对 象 ” 与 它们 也 存在 共通 的 地 方 ， 它 们 
都 有 自己 的 特征 和 行为 。 


Alan Kay 总 结 了 Smalltalk 的 五 大 基本 特征 。 这 是 第 一 种 成 功 的 面向 对 象 






























































程序 设计 语言 ， 也 是 Java 的 基础 语言 。 通 过 这 些 特征 ， 我 们 可 理解 " 纯 
粹 ”的 面向 对 象 程序 设计 方法 是 什么 样 的 : 


(1) 所 有 东西 都 是 对 象 。 可 将 对 象 想象 成 一 种 新 型 变量 ， 它 保存 着 数 
据 ， 但 可 要 求 它 对 自身 进行 操作 。 理 论 上 讲 ， 可 从 要 解决 的 问题 身上 提 
出 所 有 概念 性 的 组 件 ， 然 后 在 程序 中 将 其 表达 为 一 个 对 象 。 


(2) 程序 是 一 大 堆 对 象 的 组 合 ， 通 过 消息 传递 ， 各 对 象 知道 自己 该 做 些 
什么 。 为 了 同 对 象 发 出 请 求 ， 需 同 那 个 对 象 “ 发 送 一 条 消 轧 ”。 更 具体 地 
讲 ， 可 将 消息 想象 为 一 个 调用 请 求 ， 它 调用 的 是 从 属于 目标 对 象 的 一 个 
子 例 程 或 函数 。 


(3) 每 个 对 象 都 有 目 己 的 存储 空间 ， 可 容纳 其 他 对 象 。 或 者 说 ， 通 过 封 
装 现 有 对 象 ， 可 制作 出 新 型 对 象 。 所 以 ， 尽 管 对 象 的 概念 非常 简单 ， 但 
在 程序 中 却 可 达到 任意 高 的 复杂 程度 。 


(4) ”每 个 对 象 都 有 一 种 类 型 。 根 据 语 法 ， 每 个 对 象 都 是 茶 个 “类 ”的 一 
个 “实例 ”。 其 中 , “K” (Class) 是 “类 型 ”(Type) 的 同义词 。 一 个 类 最 
重要 的 特征 就 是 “能 将 什么 消 轧 发 给 它 ? ” 


(5) 同一 类 所 有 对 象 都 能 接收 相同 的 消息 。 这 实际 是 别 有 含 义 的 一 种 说 
法 ， 大 家 不 久 便 能 理解 。 由 于 类 型 为 “ 圆 ” (Circle) 的 一 个 对 象 也 属于 
KAI IAR” (Shape) 的 一 个 对 象 ， 所 以 一 个 圆 完全 能 接收 形状 消 

上 。 这 意味 着 可 让 程序 代码 统一 指挥 "形状 >， 令 其 自动 控制 所 有 符 

合 “形状 ?描述 的 对 象 ， 其 中 自然 包括 “* 圆 ?>。 这 一 特性 称 为 对 象 的 * 可 蔡 
换 性 ”， 是 OOP 最 重要 的 概念 之 一 。 


一 些 语言 设计 者 认为 面 癌 对 象 的 程序 设计 本 身 并 不 足以 方便 解决 所 有 形 
提倡 将 不 同 的 方法 组 合成 “多 形 程序 设计 语言 ”〈 注 释 






































©: 参见 Timothy Budd 编 著 的 《Multiparadigm Programming in Leda) , 
Addison-Wesley 1995 年 出 版 。 


1.2 对 象 的 接口 


亚 里 士 多 德 或 许 是 认真 研究 “类 型 ”概念 的 第 一 人 ， 他 曾 谈 及 “ 鱼 类 和 乌 
人 
一 个 概念 : 


所 有 对 象 一 一 尽管 各 有 特色 都 属于 某 一 系列 对 象 的 一 部 分 ， 这 些 对 
象 具 有 通用 的 特征 和 行为 。 在 Simula-67 中 ， 首 次 用 到 了 dass 这 个 关键 
a er eee erent 
AEQ)) 。 


©: 有 些 人 进行 了 进一步 的 区 分 ， 他 们 强调 “类 型 "决定 了 接口 ， 
而 “类 ?是 那个 接口 的 一 种 特殊 实现 方式 。 


Simula 是 一 个 很 好 的 例子 。 正 如 这 个 名 字 所 蜡 示 的 ， 它 的 作用 是 “ 模 

WW” (Simulate) 象 “ 银 行 出 纳 员 ”这 样 的 经 典 问 题 。 在 这 个 例子 里 ， 我 们 
A ASNT, BP. URS ARRAS. BRM MA) 部 具有 一 
些 通 用 的 特征 : 每 个 帐号 都 有 一 定 的 余额 ， 每 名 出 纳 都 能 接收 客户 的 存 
ak; 等 等 。 与 此 同时 ， 每 个 成 员 都 有 自己 的 状态 ， 每 个 帐号 都 有 不 同 的 
余额 ;每 名 出 纳 都 有 一 个 名 字 。 所 以 在 计算 机 程序 中 ， 能 用 独一无二 的 
实体 分 别 表示 出 纳 员 、 客 尸 、 帐 号 以 及 交易 。 这 个 实体 便 是 “对 象 ”， 而 
T E E 

















因此 ， 在 面向 对 象 的 程序 设计 中 ， 尺 管 我 们 真正 要 做 的 是 新 建 各 种 各 样 
的 数据 “类 型 ”〈Type) ， 但 几乎 所 有 面向 对 象 的 程序 设计 语言 都 采用 
了 “class” 关 键 字 。 当 您 看 到 “type” 这 个 字 的 时 候 ， 请 同时 想到 “class”; 
反之 亦 然 。 


建 好 一 个 类 后 ， 可 根据 情况 生成 许多 对 象 。 随 后 ， 可 将 那些 对 象 作为 要 
解决 问题 中 存在 的 元 素 进行 处 理 。 事 实 上 ， 当 我 们 进行 面向 对 象 的 程序 
设计 时 ， 面 临 的 最 大 一 项 挑战 性 就 是 ， 如 何在 “问题 空间 ”( 问 题 实际 存 
在 的 地 方 》 的 元 素 与 “方案 空间 ”( 对 实际 问题 进行 建 模 的 地 方 ， 如 计算 
WL) 的 元 素 之 间 建 立 理想 的 “一 对 一 "对 应 或 映射 关系 。 


如 何 利用 对 象 完成 真正 有 用 的 工作 呢 ? 必须 有 一 种 办 法 能 癌 对 象 发 出 请 




















求 ， 令 其 做 一 些 实际 的 事情 ， 比 如 完成 一 次 交易 、 在 屏幕 上 男 一 些 东西 
或 者 打开 一 个 开关 等 等 。 每 个 对 象 仅 能 接受 特定 的 请 求 。 我 们 同 对 象 发 
出 的 请 求 是 通过 它 的 “接口 ”(Interface) 定义 的 ， 对 象 的 “类 

型 ”或 “类 ” 则 规定 了 它 的 接口 形式 。“ 类 型 ”与 “接口 ”的 等 价 或 对 应 关系 是 
面 同 对 象 程 序 设计 的 基础 。 


下 面 让 我 们 以 电灯 泡 为 例 : 

















Type Name 


Interface 





Light lt = new LightQ; 
lt.on(); 


在 这 个 例子 中 ， 类 型 / 类 的 名 称 是 Light， 可 向 Light 对 象 发 出 的 请 求 包 
括 包 括 打开 (on) 、 关 闭 (off) 、 变 得 更 明亮 (brighten) 或 者 变 得 更 
MR (dim) 。 通 过 简单 地 声明 一 个 名 字 Ct) ， 我 们 为 Light 对 象 创建 
了 一 个 “句柄 ”。 然 后 用 new 关 键 字 新 建 类 型 为 Light 的 一 个 对 象 。 再 用 等 
号 将 其 赋 给 句柄 。 为 了 向 对 象 发 送 一 条 消息 ， 我 们 列 出 句柄 名 W ， 
再 用 一 个 句点 符号 O 把 它 同 消息 名 称 Om 连接 起 来 。 从 中 可 以 看 
出 ， 使 用 一 些 预先 定义 好 的 类 时 ， 我 们 在 程序 里 采用 的 代码 是 非常 简单 
0 直观 的 。 











1.3 HA R HI Kae 


为 方便 后 面 的 讨论 ， 让 我 们 先 对 这 一 领域 的 从 业 人 员 作 一 下 分 类 。 从 根 
本 上 说 ， 大 致 有 两 方面 的 人 员 涉足 面 癌 对象 的 编程 :“ 关 创建 者 ”创建 
新 数据 类 型 的 人 ) 以 及 “客户 程序 员 ”《〈 在 目 己 的 应 用 程序 中 采用 现成 数 
据 类 型 的 人 ;注释 由 ) 。 对 客户 程序 员 来 讲 ， 最 主要 的 目标 就 是 收集 一 
个 充斥 着 各 种 类 的 编程 < 工具 箱 ”， 以 便 快 速 开发 符合 目 己 要 求 的 应 用 。 
而 对 类 创建 者 来 说， 他们 的 目标 则 是 从 头 构建 一 个 类 ， 只 辐 客 户 程序 员 
开放 有 必要 开放 的 东西 〈 接 口 ) ， 其 他 所 有 细节 都 隐藏 起 来 。 为 什么 要 
这 样 做 ? 隐藏 之 后 ， 客 户 程序 员 惑 不 能 接触 和 改变 那些 细节 ， 上 所 以 原创 
者 不 用 担心 目 己 的 作品 会 受到 非法 修改 ， 可 确保 它们 不 会 对 其 他 人 造成 


影响。 
D: 感谢 我 的 朋友 Scott Meyers， 是 他 帮 我 起 了 这 个 名 字 。 


“fe” (Interface) 规定 了 可 对 一 个 特定 的 对 象 发 出 哪些 请 求 。 然 而 ， 
必须 在 某 个 地 方 存在 着 一 些 代码 ， 以 便 满足 这 些 请 求 。 这 些 代 码 与 那些 
隐藏 起 来 的 数据 便 叫 作 *“ 隐 藏 的 实现 ”。 站 在 程式 化 程序 编写 
(Procedural Programming) 的 角度 ， 整 个 问题 并 不 显得 复杂 。 一 种 类 型 
含有 与 每 种 可 能 的 请 求 关联 起 来 的 函数 。 一 旦 向 对 象 发 出 一 个 特定 的 请 
求 ， 就 会 调用 那个 函数 。 我 们 通常 将 这 个 过 程 总 结 为 同 对 象 “ 发 送 一 条 
消息 ”《〈 提 出 一 个 请 求 ) 。 对 象 的 职责 就 是 决定 如 何 对 这 条 消息 作出 反 
应 《执行 相应 的 代码 ) 。 


对 于 任何 关系 ， 重 要 一 点 是 让 牵连 到 的 所 有 成 员 都 遭 守 相 同 的 规则 。 创 
娃 一 个 库 时 ， 相 当 于 同 客 户 程 序 员 建立 了 一 种 关系 。 对 方 也 是 程序 员 ， 
a a a 
JE. 


在任 何人 都 能 使 用 一 个 类 的 所 有 成 员 ， 那 么 客户 程序 员 可 对 那个 类 做 任 
何事 情 ， 没 有 办 法 强制 他 们 遵守 任何 约束 。 即 便 非常 不 愿 客户 程序 员 直 
接 操 作 类 内 包含 的 一 些 成 员 ， 但 倘 耕 未 进行 访问 控制 ， 束 没有 办 法 阻止 
这 一 情况 的 发 生 一 一 所 有 东西 都 会 骏 露 无 遗 。 


有 两 方面 的 原因 促使 我 们 控制 对 成 员 的 访问 。 第 一 个 原因 是 防止 程序 员 
接触 他 们 不 该 接触 的 东西 一 一 通常 是 内 部 数据 类 型 的 设计 思想 。 耕 只 是 


















































为 了 解决 特定 的 问题 ， 用 户 只 需 操作 接口 即 可 ， 毋 需 明 白 这 些 信 息 。 我 
们 占用 户 提 供 的 实际 是 一 种 服务 ， 因 为 他 们 很 容易 就 可 看 出 哪些 对 自己 
非常 重要 ， 以 及 哪些 可 忽略 不 计 。 


进行 访问 控制 的 第 二 个 原因 是 允许 库 设 计 人 员 修 改 内 部 结构 ， 不 用 担心 
它 会 对 客户 程序 员 造 成 什么 影响 。 例 如 ， 我 们 最 开始 可 能 设计 了 一 个 形 
式 简单 的 类 ， 以 便 简 化 开发 。 以 后 又 决定 进行 改写 ， 使 其 更 快 地 运行 。 
右 接 口 与 实现 方法 早已 隔离 开 ， 并 分 别 受到 保护 ， 就 可 放心 做 到 这 一 
点 ， 只 要 求 用 户 重 新 链接 一 下 即 可 。 


Java 采 用 三 个 显 式 〈 明 确 ) 关键 字 以 及 一 个 隐 式 《暗示 ) 关键 字 来 设置 
类 边界 : public, private, protectedb\ RHA VEN friendly. AAs H MATE RE 
其 他 关键 字 ， 则 默认 为 后 者 。 这 些 关 键 字 的 使 用 和 含义 都 是 相当 直观 
的 ， 它 们 决定 了 谁 能 使 用 后 续 的 定义 内 容 。“public”( 公 共 〉 意味 着 后 
续 的 定义 任何 人 均 可 使 用 。 而 在 另 一 方面 ，“private”( 私 有 ) 意味 着 除 
您 自己 、 类 型 的 创建 者 以 及 那个 类 型 的 内 部 函数 成 员 ， 其 他 任何 人 都 不 
能 访问 后 续 的 定义 信息 。private 在 您 与 客户 程序 员 之 间 竖 起 了 一 堵 墙 。 
和 若 有 人 试图 访问 私有 成 员 ， 融 会 得 到 一 个 编译 期 错误 。 “friendly” (Ace 
的 ) 涉及 “包装 ?或 “封装 ”(Package) 的 概念 即 Java 用 来 构建 库 的 方 
法 。 耕 茶 样 东 西 是 “友好 的 "， 意 味 着 它 只 能 在 这 个 包装 的 范围 内 使 用 
《所 以 这 一 访问 级 别 有 时 也 叫 作 “包装 访问 ”) 。“protected”( 受 保护 
的 ) 与 “private” 相 似 ， 只 是 一 个 继承 的 类 可 访问 受 保 护 的 成 员 ， 但 不 能 
访问 私有 成 员 。 继 承 的 问题 不 久 就 要 谈 到 。 


























1.4 方案 的 重复 使 用 


创建 并 测试 好 一 个 类 后 ， 它 应 〈 从 理想 的 角度 ) 代表 一 个 有 用 的 代码 单 
位 。 但 并 不 象 许多 人 和 希望 的 那样 ， 这 种 重复 使 用 的 能 力 并 不 容易 实现 ; 
aaa a a 
能 用 。 


许多 人 认为 代码 或 设计 方案 的 重复 使 用 是 面向 对 象 的 程序 设计 提供 的 最 
伟大 的 一 种 杠杆 。 


为 重复 使 用 一 个 类 ， 最 简单 的 办 法 是 仅 直 接 使 用 那个 类 的 对 象 。 但 同时 
也 能 将 那个 类 的 一 个 对 象 置 入 一 个 新 类 。 我 们 把 这 叫 作 “创建 一 个 成 员 
对 象 ?。 新 类 可 由 任意 数量 和 类 型 的 其 他 对 象 构成 。 无 论 如 何 ， 只 要 新 
类 达到 了 设计 要 求 即 可 。 这 个 概念 叫 作 “组 织 ” 一 一 在 现 有 类 的 基础 上 组 
织 一 个 新 类 。 有 了 时， 我 们 也 将 组 织 称 作 “ 包 含 ” 天 系 ， 比 如 “一 辆 车 包含 
了 一 个 变速 箱 ”。 


对 象 的 组 织 具 有 极 大 的 灵活 性 。 新 类 的 “成 员 对 象 ”通常 设 为 “ 私 

A” (Private) ， 使 用 这 个 类 的 客户 程序 员 不 能 访问 它们 。 这 样 一 来 ， 
我 们 可 在 不 干扰 客户 代码 的 前 提 下 ， 从 容 地 修改 那些 成 员 。 也 可 以 

在 “运行 期 ?更 改 成 员 ， 这 进一步 增 大 了 有 灵活 性 。 后 面 要 讲 到 的 “继承 ?并 
不 有 具备 这 种 灵活 性 ， 因 为 编译 器 必须 对 通过 继承 创建 的 类 加 以 限制 。 


由 于 继承 的 重要 性 ， 所 以 在 面向 对 象 的 程序 设计 中 ， 它 经 常 锌 重点 强 
调 。 作 为 新 加 入 这 一 领域 的 程序 员 ， 或 许 早已 完 入 为 主 地 认为 “继承 应 
当 随 处 可 见 ”"。 沿 这 种 思路 产生 的 设计 将 是 非常 茶 拙 的 ， 会 大 大 增加 程 
序 的 复杂 程度 。 相 反 ， 新 建 类 的 时 候 ， 首 先 应 考虑 “组 织 ? 对 象 ， 这样 做 
显得 更 加 简单 和 灵活 。 利 用 对 象 的 组 织 ， 我 们 的 设计 可 保持 清爽 。 一 旦 
需要 用 到 继承 ， 就 会 明显 意识 到 这 一 点 。 
































1.5 继承 : 重新 使 用 接口 


就 其 本 身 来 说 ， 对 象 的 概念 可 为 我 们 带 来 极 大 的 便利 。 它 在 概念 上 允许 
我 们 将 各 式 各 样 数据 和 功能 封装 到 一 起 。 这 样 便 可 恰当 表达 “问题 空 
间 * 的 概念 ， 不 用 刻意 遵照 基础 机 器 的 表达 方式 。 在 程序 设计 语言 中 ， 
这 些 概念 则 反映 为 具体 的 数据 类 型 〈 使 用 qlass 关 键 字 ) 。 


我 们 费 尽心 思 做 出 一 种 数据 类 型 后 ， 假 如 不 得 不 又 新 建 一 种 类 型 ， 令 其 
实现 大 致 相同 的 功能 ， 那 会 是 一 件 非常 令 人 灰心 的 事情 。 但 知 能 利用 现 
成 的 数据 类 型 ， 对 其 进行 “克隆 ”， 再 根据 情况 进行 添加 和 修改 ， 情 况 就 
显得 理想 多 了 。 “继承 ? 正 是 针对 这 个 目标 而 设计 的 。 但 继承 并 不 完全 等 
价 于 克隆 。 在 继承 过 程 中 ， 知 原始 类 〈 正 式 名 称 叫 作 基础 类 、 超 类 或 父 
X) 发 生 了 变化 ， 修 改过 的 “克隆 ”类 “〈 正 式 名 称 叫 作 继 承 类 或 者 子 类 ) 
也 会 反映 出 这 种 变化 。 在 Java 语 言 中 ， 继 承 是 通过 extends 关 键 字 实现 的 


使 用 继承 时 ， 相 当 于 创建 了 一 个 新 类 。 这 个 新 类 不 仅 包 含 了 现 有 类 型 的 
所 有 成 员 〔( 尽 管 private 成 员 被 隐藏 起 来 ， 且 不 能 访问 ) ， 但 更 重要 的 
是 ， 它 复制 了 基础 类 的 接口 。 也 束 是 说 ， 可 癌 基 础 类 的 对 象 友 送 的 所 有 
消息 亦 可 原样 发 给 衍生 类 的 对 象 。 根 据 可 以 及 送 的 消 恩 ， 我 们 能 知道 类 
的 类 型 。 这 意味 着 衍生 类 具有 与 基础 类 相同 的 类 型 ! 为 真正 理解 面向 对 
象 程序 设计 的 含义 ， 首 先 必 须 认 识 到 这 种 类 型 的 等 价 关 系 。 


由 于 基础 类 和 衍生 类 具有 相同 的 接口 ， 所 以 那个 接口 必须 进行 特殊 的 设 
计 。 也 惑 是 说 ， 对 象 接 收 到 一 条 特定 的 消 轧 后 ， 必 须 有 一 个 “方法 ?能够 
执行 。 各 只 是 简单 地 继承 一 个 类 ， 并 不 做 其 他 任何 事情 ， 来 目 基 础 类 接 
口 的 方法 就 会 直接 照搬 到 衍生 类 。 这 意味 着 衍生 类 的 对 象 不 仅 有 相同 的 
类 型 ， 也 有 同样 的 行为 ， 这 一 后 果 通 常 是 我 们 不 愿 见 到 的 。 


有 了 两 种 做 法 可 将 新 得 的 衍生 类 与 原来 的 基础 类 区 分 开 。 第 一 种 做 法 十 分 
简单 : 为 衍生 类 添加 新 函数 〔 功 能 ) 。 这 些 新 函数 并 非 基 础 类 接口 的 一 
部 分 。 进 行 这 种 处 理 时 ， 一 般 都 是 意识 到 基础 类 不 能 满足 我 们 的 要 求 ， 
所 以 需要 添加 更 多 的 函数 。 这 是 一 种 最 简单 、 最 基本 的 继承 用 法 ， 大 多 
数 时 候 都 可 完美 地 解决 我 们 的 问题 。 然 而 ， 事 先 还 是 要 仔细 调查 目 己 的 
基础 关 是 人 否 真 的 需要 这 些 额 外 的 函数 。 


1.5.1 改善 基础 类 



































尽管 extends 关 键 字 上 暗示 独 我 们 要 为 接口 “扩展 ?新 功能 ， 但 实情 并 非 肯定 
如 此 。 为 区 分 我 们 的 新 类 ， 第 二 个 办 法 是 改变 基础 类 一 个 现 有 函数 的 行 
为 。 我 们 将 其 称 作 * 改 善 ? 那 个 函数 。 


为 改善 一 个 函数 ， 只 需 为 衍生 类 的 函数 建立 一 个 新 定义 即 可 。 我 们 的 目 
标 是 :“ 尽 省 使 用 的 函数 接口 未 变 ， 但 它 的 新 版 本 具有 不 同 的 表现 ”。 


1.5.2 等 价 与 类 似 关 系 


针对 继承 可 能 会 产生 这 样 的 一 个 争论 : 继承 只 能 改善 原 基 础 类 的 函数 
吗 ? 知 答 案 是 肯定 的 ， 则 衍生 类 型 就 是 与 基础 类 完全 相同 的 类 型 ， 因 为 
都 拥有 完全 相同 的 接口 。 这 样 造 成 的 结果 就 是 : 我 们 完全 能 够 将 衍生 类 
的 一 个 对 象 换 成 基础 类 的 一 个 对 象 ! 可 将 其 想象 成 一 种 “ 纯 蔡 换 ”。 在 某 
种 意义 上 ， 这 是 进行 继承 的 一 种 理想 方式 。 此 时 ， 我 们 通常 认为 基础 类 
和 衍生 类 之 间 存 在 一 种 “等 价 ” 关 系 一 一 因为 我 们 可 以 理直气壮 地 

说 :“ 圆 就 是 一 种 几何 形状 >。 为 了 对 继承 进行 测试 ， 一 个 办 法 就 是 看 看 
目 己 是 否 能 把 它们 套 入 这 种 “等 价 " 关 系 中 ， 看 看 是 否 有 意义 。 


但 在 许多 时 候 ， 我 们 必须 为 衍生 类 型 加 入 新 的 接口 元 素 。 所 以 不 仅 扩展 
了 接口 ， 也 创建 了 一 种 新 类 型 。 这 种 新 类 型 仍 可 符 换 成 基础 类 型 ， 但 这 
种 人 答 换 并 不 是 完美 的 ， 因 为 不 可 在 基础 类 里 访问 新 函数 。 我 们 将 其 称 
作 * 类 似 ? 关 系 ; 新 类 型 拥有 旧 类 型 的 接口 ， 但 也 包含 了 其 他 函数 ， 所 以 
不 能 说 它们 是 完全 等 价 的 。 举 个 例子 来 说 ， 让 我 们 考虑 一 下 制冷 机 的 情 
况 。 假 定 我 们 的 房间 连 好 了 用 于 制冷 的 各 种 控制 项 ， 也 就 是 说， 我 们 已 
拥有 必要 的 “接口 "来 控制 制冷 。 现 在 假设 机 器 出 了 故障 ， 我 们 把 它 换 成 
一 台新 型 的 冷 、 热 两 用 空调 ， 冬 天 和 夏天 均 可 使 用 。 冷 、 热 空调 “类 
似 ” 制 冷 机 ， 但 能 做 更 多 的 事情 。 由 于 我 们 的 房间 只 安装 了 控制 制冷 的 
设备 ， 所 以 它们 只 限于 同 新 机 器 的 制冷 部 分 打交道 。 新 机 器 的 接口 已 得 
到 了 扩展 ， 但 现 有 的 系统 并 不 知道 除 原 始 接口 以 外 的 任何 东西 。 


认识 了 等 价 与 类 似 的 区 别 后 ， 再 进行 替换 时 就 会 有 把 握 得 多 。 尽 管 大 多 
数 时 候 < 纯 蔡 换 "已 经 足够 ， 但 您 会 发 现在 某 些 情况 下 ， 仍 然 有 明显 的 理 
由 需要 在 衍生 类 的 基础 上 增添 新 功能 。 通 过 前 面 对 这 两 种 情况 的 讨论 ， 
相信 大 家 已 心中 有 数 该 如 何 做 。 





















































1.6 多 形 对 象 的 互 换 使 用 


通常 ， 继 承 最 终 会 以 创建 一 系列 类 收场 ， 所 有 类 都 建立 在 统一 的 接口 基 
础 上 。 我 们 用 一 幅 颠 倒 的 树 形 图 来 阐明 这 一 点 注释 @@》: 


©: 这儿 采用 了 “统一 记号 法 ”， 本 书 将 主要 采用 这 种 方法 。 
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对 这 样 的 一 系列 类 ， 我 们 要 进行 的 一 项 重要 处 理 就 是 将 衍生 类 的 对 象 当 
作 基 础 闫 的 一 个 对 象 对 竺 。 REAR BEA, B's Beka aa 
再 编写 单一 的 代码 ， 令 其 忽略 类 型 的 特定 细节 ， 只 与 基础 类 打交道 。 这 
样 一 来 ， 那 些 代 码 束 可 与 类 型 信息 分 开 。 所 以 更 易 编 写 ， 也 更 易 理解 。 

此 外 ， 寿 通过 继承 增添 了 一 种 新 类 型 ， 如 “三 角形 ”， 那么 我们 为 “几何 

形状 ”新 类 型 编写 的 代码 会 象 在 旧 类 型 里 一 样 民 好 地 工作 。 所 以 说 程序 
具备 了 “扩展 能 力 ”” 具有 “扩展 性 ” 


以 上 面 的 例子 为 基础 ， 假 设 我 们 用 Java 写 了 这 样 一 个 函数 : 


void doStuff(Shape s) { 





s.erase(); 
a rere 


s.draw(); 


这 个 函数 可 与 任何 “几何 形状 ”(Shape) 通信 ， 所 以 完全 独立 于 它 要 描 
绘 (draw) 和 删除 Cerase) 的 任何 特定 类 型 的 对 象 。 如 果 我 们 在 其 他 一 
些 程序 里 使 用 doStuffO 函 数 : 


Circle c = new Circle(); 


Triangle t = new Triangle(); 
Line 1 = new Line(); 

doStuff (c); 

doStuff (t); 


doStuff (1); 
那么 对 doStuffO 的 调用 会 自动 良好 地 工作 ， 无 论 对 象 的 具体 类 型 是 什 
Zo 


这 实际 是 一 个 非常 有 用 的 编程 技巧 。 请 考虑 下 面 这 行 代码 : 
doStuff(c); 


此 时 ， 一 个 Circle〈 圆 ) 句柄 传递 给 一 个 本 来 期 待 Shape《〈 形 状 ) 句柄 的 
函数 。 由 于 圆 是 一 种 几何 形状 ， 所 以 doStuffO 能 正确 地 进行 处 理 。 也 恕 
是 说 ， 凡 是 doStuffO 能 发 给 一 个 Shape 的 消息 ，Circle 也 能 接收 。 所 以 这 
样 做 是 安全 的 ， 不 会 造成 错误 。 


我 们 将 这 种 把 衍生 类 型 当 作 它 的 基本 类 型 处 理 的 过 程 叫 

作 *“Upcasting”( 上 调 造 型 ) 。 其 中 , “cast"”《〈 造 型 ) 是 指 根据 一 个 现成 
的 模型 创建 ， 而 “Up”( 向 上 ) 表明 继承 的 方向 是 从 “上 面 * 来 的 一 一 即 基 
础 类 位 于 顶部 ， 而 衍生 类 在 下 方 展开 。 上 所 以 ， 根 据 基础 类 进行 造型 就 是 
一 个 从 上 面 继承 的 过 程 ， 即 “Upcasting”。 


在 面 癌 对 象 的 程序 里 ， 通 常 都 要 用 到 上 漳 造 型 技术 。 这 是 避免 去 调查 准 








确 类 型 的 一 个 好 办 法 。 请 看 看 doStuffO 里 的 代码 : 
s.erase(); 

Wes 

s.draw(); 


注意 它 并 未 这 样 表达 : “如 果 你 是 一 个 Circle， 就 这 样 做 ， 如 果 你 是 一 个 
Square， 就 那样 做 ;等 等 "。 硅 那样 编写 代码 ， 束 需 检 查 一 个 Shape 所 有 
可 能 的 类 型 ， 如 圆 、 和 矩形 等 等 。 这 显然 是 非常 腑 烦 的 ， 而 且 每 次 添加 了 
一 种 新 的 Shape 类 型 后 ， 都 要 相应 地 进行 修改 。 在 这 儿 ， 我 们 只 需 

说 : “你 是 一 种 几何 形状 ， 我 知道 你 能 将 自己 删 摊 ， 即 erase0; te A OK 
取 那 个 行动 ， 并 上 自己 去 控制 所 有 的 细节 吧 。” 


1.6.1 动态 绑 定 


在 doStuffO 的 代码 里 ， 最 让 人 吃惊 的 是 尽管 我 们 没 作 出 任何 特殊 指示 ， 
采取 的 操作 也 是 完全 正确 和 恰当 的 。 我 们 知道 ， 为 Circle 调 用 drawO 时 执 
行 的 代码 与 为 一 个 Square 或 Line 调 用 drawO 时 执行 的 代码 是 不 同 的 。 但 
在 将 draw0 消 息 发 给 一 个 匿名 Shape 时 ， 根 据 Shape 句 柄 当时 连接 的 实际 
类 型 ， 会 相应 地 采取 正确 的 操作 。 这 当然 令 人 人 惊讶， 因为 当 Java 编 译 器 
为 doStuffO 编 译 代码 时 ， 它 并 不 知道 自己 要 操作 的 准确 类 型 是 什么 。 尽 
管 我 们 确实 可 以 保证 最 终 会 为 Shape 调 用 erase()， 为 Shape 调 用 draw0)， 

但 并 不 能 保证 为 特定 的 Circle，Square 或 者 Line 调 用 什么 。 然 而 最 后 采取 
的 操作 同样 是 正确 的 ， 这 是 怎么 做 到 的 呢 ? 


将 一 条 消息 发 给 对 象 时 ， 如 有 果 并 不 知道 对 方 的 具体 类 型 是 什么 ， 但 采取 
的 行动 同样 是 正确 的 ， 这 种 情况 就 叫 作 “多 形 性 ”(Polymorphism) 。 对 
面 问 对 象 的 程序 设计 语言 来 说 ， 它 们 用 以 实现 多 形 性 的 方法 叫 作 "动态 

绑 定 ”。 编 译 器 和 运行 期 系统 会 负责 对 所 有 细 贡 的 控制 ， 我 们 只 需 知道 

会 发 生 什 么 事情 ， 而 且 更 重要 的 是 ， 如 何 利用 它 帮 助 目 己 设计 程序 。 


有 些 语言 要 求 我 们 用 一 个 特殊 的 关键 字 来 允许 动态 绑 定 。 在 C++ 中 ， 这 

个 关键 字 是 virtual。 在 Java 中 ， 我 们 则 完全 不 必 记 住 添 加 一 个 关键 字 ， 

因为 函数 的 动态 绑 定 是 自动 进行 的 。 所 以 在 将 一 条 消息 发 给 对 象 时 ， 我 
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1.6.2 抽象 的 基础 类 和 接口 


设计 程序 时 ， 我 们 经 第 都 希望 基础 类 只 为 自己 的 衍生 类 提供 一 个 接口 。 
也 就 是 说 ， 我 们 不 想 其 他 任何 人 实际 创建 基础 类 的 一 个 对 象 ， 只 对 上 济 
造型 成 它 ， 以 便 使 用 它们 的 接口 。 为 达到 这 个 目的 ， 需 要 把 那个 类 变 
成 “抽象 ”的 一 一 使 用 abstract 关 键 字 。 和 大 有 人 试图 创建 抽象 类 的 一 个 对 
象 ， 编 译 器 就 会 阻止 他 们 。 这 种 工具 可 有 效 强制 实行 一 种 特殊 的 设计 。 


亦 可 用 abstract 关 键 字 描述 一 个 尚未 实现 的 方法 一 一 作为 一 个 “ 根 ” 使 用 ， 
指出 :“ 这 是 适用 于 从 这 个 类 继承 的 所 有 类 型 的 一 个 接口 函数 ， 但 目前 
尚 没有 对 它 进行 任何 形式 的 实现 。” 抽 象 方法 也 许 只 能 在 一 个 抽象 类 里 
创建 。 继 承 了 一 个 类 后 ， 那 个 方法 就 必须 实现 ， 否 则 继承 的 类 也 会 变 
成 “抽象 类。 通过 创建 一 个 抽象 方法 ， 我 们 可 以 将 一 个 方法 置 入 接口 
中 ， 不 必 再 为 那个 方法 提供 可 能 坚 无 意义 的 主体 代码 。 


interface 〈 接 口 ) 关键 字 将 抽象 类 的 概念 更 延伸 了 一 步 ， 它 完全 禁止 了 
所 有 的 函数 定义 。*“ 接 口 ?是 一 种 相当 有 效 和 和 用 的 工具 。 另 外 如 果 上 自己 
愿意 ， 亦 可 将 多 个 接口 都 合并 到 一 起 〈 不 能 从 多 个 普通 class 或 abstract 
class 中 继承 ) 。 











1.7 对 象 的 创建 和 存在 时 间 


从 技术 角度 说 ，OOP《〈 面 癌 对 象 程序 设 计 ) 只 是 涉及 抽象 的 数据 类 型 、 
继承 以 及 多 形 性 ， 但 另 一 些 问题 也 可 能 显得 非常 重要 。 本 节 将 就 这 些 问 
题 进行 探讨 。 


最 重要 的 问题 之 一 是 对 象 的 创建 及 破坏 方式 。 对 象 需要 的 数据 位 于 哪 

儿 ， 如 何 控制 对 象 的 “存在 时 间 ” 呢 ?针对 这 个 问题 ， 解 决 的 方案 是 各 异 
其 趣 的 。C++ 认 为 程序 的 执行 效率 是 最 重要 的 一 个 问题 ， 所 以 它 允 许 程 
序 员 作出 选择 。 为 获得 最 快 的 运行 速度 ， 存 储 以 及 存在 时 间 可 在 编写 程 
序 时 决定 ， 只 需 将 对 象 放置 在 堆栈 〈 有 时 也 叫 作 目 动 或 定 域 变 量 ) 或 者 
静态 存储 区 域 即 可 。 这 样 便 为 存储 空间 的 分 配 和 释放 提供 了 一 个 优先 

级 。 茶 些 情况 下 ， 这 种 优先 级 的 控制 是 非常 有 价值 的 。 然 而 ， 我 们 同时 
也 牺牲 了 灵活 性 ， 因 为 在 编写 程序 时 ， 必 须知 道 对 象 的 准确 的 数量 、 存 
在 时 间 、 以 及 类 型 。 如 果 要 解决 的 是 一 个 较 常规 的 问题 ， 如 计算 机 辅助 
设计 、 仓 储 管理 或 者 空中 交通 控制 ， 这 一 方法 就 显得 太 局 限 了 。 


第 二 个 方法 是 在 一 个 内 存 池 中 动态 创建 对 象 ， 该 内 存 池 亦 叫 “ 堆 ? 或 

者 “内 存 堆 ?。 知 采用 这 种 方式 ， 除 非 进 入 运行 期 ， 否 则 根本 不 知道 到 底 
需要 多 少 个 对 象 ， 也 不 知道 它们 的 存在 时 间 有 多 长 ， 以 及 准确 的 类 型 是 
什么 。 这 些 参数 都 在 程序 正式 运行 时 才 决 定 的 。 若 需 一 个 新 对 象 ， 只 需 
在 需要 它 的 时 候 在 内 存 堆 里 简单 地 创建 它 即 可 。 由 于 存储 空间 的 管理 是 
运行 期 间 动态 进行 的 ， 所 以 在 内 存 堆 里 分 配 存储 空间 的 时 间 比 在 堆栈 里 
创建 的 时 间 长 得 多 在 堆栈 里 创建 存储 空间 一 般 只 需要 一 个 简单 的 指 
令 ， 将 堆栈 指针 向 下 或 向 下 移动 即 可 〉。 由 于 动态 创建 方法 使 对 象 本 来 
就 倾 问 于 复杂 ， 上 所 以 碍 找 存储 空间 以 及 释放 和 它 所 需 的 额外 开销 不 会 为 对 
象 的 创建 造成 明显 的 影响 。 除 此 以 外 ， 更 大 的 灵活 性 对 于 常规 编程 问题 
的 解决 是 至 关 重 要 的 。 


C++ 人 允许 我 们 决定 是 在 写 程序 时 创建 对 象 ， 还 是 在 运行 期 间 创 建 ， 这 种 
控制 方法 更 加 灵活 。 大 家 或 许 认 为 既然 它 如 此 灵活 ， 那 么 无 论 如 何 都 应 
在 内 存 堆 里 创建 对 象 ， 而 不 是 在 堆栈 中 创建 。 但 还 要 考虑 为 外 一 个 问 
题 ， 亦 即 对 象 的 “存在 时 间 ” 或 者 “生存 时 间 ”(Lifetime〉。 寿 在 堆栈 或 
者 静态 存储 空间 里 创建 一 个 对 象 ， 编 译 器 会 判断 对 象 的 持续 时 间 有 多 
长 ， 到 时 会 自动 “破坏 ”或 者 “清除 ” 它 。 程 序 员 可 用 两 种 方法 来 破坏 一 个 
HR: 用 程序 化 的 方式 决定 何 时 破坏 对 象 ， 或 者 利用 由 运行 环境 提供 的 










































































一 种 “垃圾 收集 器 特性， 自动 寻找 那些 不 再 使 用 的 对 象 ， 并 将 其 清除 。 

当然 ， 垃 圾 收集 器 显得 方便 得 多 ， 但 要 求 所 有 应 用 程序 都 必须 容忍 垃圾 
收集 器 的 存在 ， 并 能 默许 随 垃 圾 收集 带 来 的 额外 开销 。 但 这 并 不 符合 

C++ 语言 的 设计 宗旨 ， 所 以 未 能 包括 到 C++ 里 。 但 Java 确 实 提供 了 一 个 
垃圾 收集 器 (Smalltalk 也 有 这 样 的 设计 ; 尽管 Delphi 默 认为 没有 垃圾 收 
集 器 ， 但 可 选择 安装 ;而 C++ 亦 可 使 用 一 些 由 其 他 公司 开发 的 垃圾 收集 
产品 ) 。 


本 节 剩 下 的 部 分 将 讨论 操纵 对 象 时 要 考虑 的 另 一些 因 素 。 
1.7.1 RAH AK aS 


针对 一 个 特定 问题 的 解决 ， 如 末 事 先 不 知道 需要 多 少 个 对 象 ， 或 者 它们 
的 持续 时 间 有 多 长 ， 那 么 也 不 知道 如 何 保存 那些 对 象 。 既 然 如 此 ， 怎 样 
才能 知道 那些 对 象 要 求 多 少 空间 呢 ? 事 匈 上 根本 无 法 提前 知道 ， 除 非 进 
入 运行 期 。 

只 


在 面 癌 对 象 的 设计 中 ， 大 多 数 问题 的 解决 办 法 似乎 都 有 些 轻率 \ 是 
简单 地 创建 吃 一 种 类 型 的 对 象 。 用 于 解决 特定 问题 的 新 型 对 象 容纳 了 指 
回 其 他 对 象 的 色 柄 。 当 然 ， 也 可 以 用 数组 来 做 同样 的 事情 ， 那 是 大 多 数 
语言 都 具有 的 一 种 功能 。 但 不 能 只 看 到 这 一 点 。 这 种 新 对 象 通常 叫 

作 ”* 集 合 ”〈 亦 叫 作 一 个 “容器 ?”， 但 AWIT 在 不 同 的 场合 应 用 了 这 个 术语 ， 
所 以 本 书 将 一 直 沿 用 “集合 ”的 称呼 。 在 需要 的 时 候 ， 集 合 会 自动 扩充 自 
己 ， 以 便 适 应 我 们 在 其 中 置 入 的 任何 东西 。 所 以 我 们 事先 不 必 知 道 要 在 
Be re re mre et eee 
责 好 了 。 


六 运 的 是 ， 设 计 优 恨 的 OOP 语 言 都 配套 提供 了 一 系列 集合 。 在 C++ 中 ， 

它们 是 以 “标准 模板 库 ”(STL) 的 形式 提供 的 。Object Pascal 用 自己 
的 “可 视 组 件 库 ”(VCL) 提供 集合 。Smalltalk 提 供 了 一 套 非常 完整 的 集 
合 。 而 Java 也 用 自己 的 标准 库 提供 了 和 集合。 在 某 些 库 中 ， 一 个 常规 集合 
便 可 满足 人 们 的 大 多 数 要 求 ， 而 在 男 一 些 库 中 (特别 是 C++ 的 库 ) ， 则 
面 同 不 同 的 需求 提供 了 不 同类 型 的 集合 。 例 如 ， 可 以 用 一 个 矢量 统一 对 
所 有 元 素 的 访问 方式 ; 一 个 链接 列表 则 用 于 保证 所 有 元 素 的 插入 统一 。 

所 以 我 们 能 根据 自己 的 需要 选择 适当 的 类 型 。 其 中 包括 集 、 队 列 、 散 列 
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所 有 集合 都 提供 了 相应 的 读 写 功能 。 将 某 样 东 西 置 入 集合 时 ， 采 用 的 方 

















式 是 十 分 明显 的 。 有 一 个 叫 作 * 推 ”(Push) ~ “W” Cadd) 或 其 他 类 
似 名 字 的 函数 用 于 做 这 件 事 情 。 但 将 数据 从 集合 中 取出 的 时 候 ， 方 式 却 
并 不 总 是 那么 明显 。 如 果 是 一 个 数组 形式 的 实体 ， 比 如 一 个 矢量 
(Vector) ， 那 么 也 许 能 用 索引 运算 符 或 函数 。 但 在 许多 情况 下 ， 这 样 
做 往往 会 无 功 而 返 。 此 外 ， 单 选 定 函数 的 功能 是 非常 有 限 的 。 如 果 想 对 
全 而 不 是 仅仅 面 癌 一 个 ， 这 时 又 访 
iba ZZ Ne ? 


办 法 就 是 使 用 一 个 “继续 器 ”(Iterator) ， 它 属于 一 种 对 象 ， 负 责 选 择 集 
合 内 的 元 素 ， 并 把 它们 提供 给 继承 器 的 用 户 。 作 为 一 个 类 ， 它 也 提供 了 
一 级 抽象 。 利 用 这 一 级 抽象 ， 可 将 集合 细 市 与 用 于 访问 那个 集合 的 代码 
隅 离开 。 通 过 继承 器 的 作用 ， 集 合 被 抽象 成 一 个 简单 的 序列 。 继 承 器 多 
许 我 们 过 历 那 个 序列 ， 同 时 毋 需 关 心 基 础 结构 是 什么 一 一 换言之 ， 不 管 
它 是 一 个 矢量 、 一 个 链接 列表 、 一 个 堆栈 ， 还 是 其 他 什么 东西 。 这 样 一 
来 ， 我 们 就 可 以 灵活 地 改变 基础 数据 ， 不 会 对 程序 里 的 代码 造成 干扰 。 
Java 最 开始 (在 1.0 和 1.1 版 中 )〉 提供 的 是 一 个 标准 继承 器 ， 名 为 
Enumeration (H) ， 为 它 的 所 有 集合 类 提供 服务 。Java 1.2 新 增 一 个 
更 复杂 的 集合 库 ， 其 中 包含 了 一 个 名 为 Iterator 的 继承 器 ， 可 以 做 比 老式 
的 Enumeration 更 多 的 事情 。 


从 设计 角度 出 发 ， 我 们 需要 的 是 一 个 全 功能 的 序列 。 通 过 对 它 的 操纵 ， 
应 该 能 解决 目 己 的 问题 。 如 果 一 种 类 型 的 序列 即 可 满足 我 们 的 所 有 要 
求 ， 那 么 完全 没有 必要 再 换 用 不 同 的 类 型 。 有 两 方面 的 原因 促使 我 们 需 
要 对 集合 作出 选择 。 首 先 ， 集 合 提供 了 不 同 的 接口 类 型 以 及 外 部 行为 。 
堆栈 的 接口 与 行为 与 队列 的 不 同 ， 而 队列 的 接口 与 行为 又 与 一 个 集 
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活性 。 


其 次 ， 不 同 的 集合 在 进行 特定 操作 时 往往 有 不 同 的 效率 。 最 好 的 例子 便 
ÆRE (Vector) 和 列表 (List〉 的 区 别 。 它 们 都 属于 简单 的 序列 ， 拥 
有 完全 一 致 的 接口 和 外 部 行为 。 但 在 执行 一 些 特定 的 任务 时 ， 需 要 的 开 
销 却 是 完全 不 同 的 。 对 矢量 内 的 元 际 进 行 的 随机 访问 〈 存 取 ) 是 一 种 各 
时 操作 ;无 论 我 们 选择 的 选择 是 什么 ， 需 要 的 时 间 量 都 是 相同 的 。 但 在 
一 个 链接 列表 中 ， 寿 想到 处 移动 ， 并 随机 挑选 一 个 元 系 ， 残 需 付出 “ 惨 
重 ” 的 代价 。 而 且 假 设 某 个 元 素 位 于 列表 较 远 的 地 方 ， 找 到 它 所 需 的 时 
间 也 会 长 许多 。 但 在 男 一 方面 ， 如 果 想 在 序列 中 部 插入 一 个 元 素 ， 用 列 
表 就 比 用 和 天 量 划 算得 多 。 这 些 以 及 其 他 操作 都 有 不 同 的 执行 效率 ， 有 具体 



































取决 于 序列 的 基础 结构 是 什么 。 在 设计 阶段 ， 我 们 可 以 先 从 一 个 列表 开 
始 。 最 后 调整 性 能 的 时 候 ， 再 根据 情况 把 它 换 成 天 量 。 由 于 抽象 是 通过 
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道 。 


最 后 ， 记 住 集合 只 是 一 个 用 来 放置 对 象 的 储藏 所 。 如 果 那 个 储藏 所 能 满 
足 我 们 的 所 有 需要 ， 就 完全 没 必要 关心 它 具 体 是 如 何 实现 的 〈 这 是 大 多 
数 类 型 对 象 的 一 个 基本 概念 ) 。 如 果 在 一 个 编程 环境 中 工作 ， 它 由 于 其 
他 因素 《比如 在 Windows 下 运行 ， 或 者 由 垃圾 收集 器 带 来 了 开销 ) 产生 
了 内 在 的 开销 ， 那 么 矢量 和 链接 列表 之 间 在 系统 开销 上 的 差异 就 或 许 不 
是 一 个 大 问题 。 我 们 可 能 只 需要 一 种 类 型 的 序列 。 甚 至 可 以 想象 有 一 

个 “完美 > 的 集合 抽象 ， 它 能 根据 自己 的 使 用 方式 上 自动 改变 基层 的 实现 方 


Tye 
1.7.2 单 根 结构 


在 面向 对 象 的 程序 设计 中 ， 由 于 C++ 的 引入 而 显得 尤为 突出 的 一 个 问题 
er 所 有 类 最 终 是 否 都 应 从 单独 一 个 基础 类 继承 。 在 Java 中 《与 其 他 几 
乎 所 有 OOP 语 言 一 样 ) ， 对 这 个 问题 的 答案 都 是 肯定 的 ， 而 且 这 个 终 级 
a 


单 根 结构 中 的 所 有 对 象 都 有 一 个 通用 接口 ， 所 以 它们 最 终 都 属于 相同 的 
类 型 。 另 一 种 方案 《就 象 C++ 那样 ) 是 我 们 不 能 保证 所 有 东西 都 属于 相 
同 的 基本 类 型 。 从 疝 后 兼容 的 角度 看 ， 这 一 方案 可 与 C 模 型 更 好 地 配 
合 ， 而 且 可 以 认为 它 的 限制 更 少 一 些 。 但 假期 我 们 想 进 行 纯粹 的 面向 对 
象 编程 ， 那 么 必须 构建 自己 的 结构 ， 以 期 获得 与 内 建 到 其 他 OOP 语 言 里 
的 同样 的 便利 。 需 添加 我 们 要 用 到 的 各 种 新 类 库 ， 还 要 使 用 为 一 些 不 兼 
容 的 接口 。 理 所 当然 地 ， 这 也 需要 付出 额外 的 精力 使 新 接口 与 自己 的 设 
计 方 案 配 合 〈 可 能 还 需要 多 重 继承 ) 。 为 得 到 C++ 额 外 的 “灵活 性 ”， 付 
出 这 样 的 代价 值得 吗 ? 当然 ， 如 果真 的 需要 一 一 如 果 早 已 是 C 专 家 ， 如 
果 对 C 有 难 售 的 情 络 一 一 那么 就 真 的 很 值得 。 但 假如 你 是 一 名 新 手 ， 首 
次 接触 这 类 设计 ， 象 Java 那 样 的 蔡 换 方案 也 许 会 更 省 事 一 些 。 


单 根 结构 中 的 所 有 对 象 〈《 比 如 所 有 Java 对 象 ) 都 可 以 保证 拥有 一 些 特 定 
的 功能 。 在 自己 的 系统 中 ， 我 们 知道 对 每 个 对 象 都 能 进行 一 些 基本 操 

作 。 一 个 单 根 结构 ， 加 上 所 有 对 象 部 在 内 存 堆 中 创建 ， 可 以 极 大 简化 参 
数 的 传递 〈 这 在 C++ 里 是 一 个 复杂 的 概念 ) 。 












































利用 单 根 结构 ， 我 们 可 以 更 方便 地 实现 一 个 垃圾 收集 上 器。 与 此 有 关 的 必 
要 文 持 可 安装 于 基础 类 中 ， 而 垃圾 收集 器 可 将 适当 的 消息 发 给 系统 内 的 
任何 对 象 。 如 果 没 有 这 种 单 根 结构 ， 而 且 系 统 遂 过 一 个 句柄 来 操纵 对 
象 ， 那 么 实现 垃圾 收集 器 的 途径 会 有 很 大 的 不 同 ， 而 且 会 面临 许多 障 


但 。 


由 于 运行 期 的 类 型 信息 肯定 存在 于 所 有 对 象 中 ， 所 以 永远 不 会 遇 到 判断 
不 出 一 个 对 象 的 类 型 的 情况 。 这 对 系统 级 的 操作 来 说 显得 特别 重要 ， 比 
如 违例 控制 ， 而 且 也 能 在 程序 设计 时 获得 更 大 的 灵活 性 。 


但 大 家 也 可 能 产生 疑问 ， 既 然 你 把 好 处 说 得 这 么 天 花 乱 附 ， 为 什么 
C++ 没 有 采用 单 根 结构 呢 ? 事 实 上 ， 这 是 早期 在 效率 与 控制 上 权衡 的 一 
种 结果 。 单 根 结构 会 带 来 程序 设计 上 的 一 些 限制 。 而 且 更 重要 的 是 ， 它 
加 大 了 新 程序 与 原 有 C 人 代码 兼容 的 难度 。 尽 管 这 些 限 制 仅 在 特定 的 场合 
会 真 的 造成 问题 ， 但 为 了 获得 最 大 的 灵活 程度 ，C++ 最 终 决 定 放 弃 采 用 
单 根 结构 这 一 做 法 。 而 Java 不 存在 上 述 的 问题 ， 它 是 全 新 设计 的 一 种 语 
言 ， 不 必 与 现 有 的 语言 保持 所 请 的 “ 回 后 羔 容 *。 所 以 很 自然 地 ， 与 其 他 
大 多 数 面 癌 对 象 的 程序 设计 语言 一 样 ， 单 根 结构 在 Java 的 设计 方案 中 很 
AIK POR » 


1.7.3 集合 库 与 方便 使 用 集合 


由 于 集合 是 我 们 经 常 都 要 用 到 的 一 种 工具 ， 所 以 一 个 集合 库 是 十 分 必要 

的 ， 它 应 该 可 以 方便 地 重复 使 用 。 这 样 一 来 ， 我 们 就 可 以 方便 地 取 用 各 

种 集合 ， 将 其 插入 自己 的 程序 。Java 提 供 了 这 样 的 一 个 库 ， 尽 管 它 在 

1.0 和 1.1 中 都 显得 非常 有 限 Clava ”1.2 的 集合 库 则 无 疑 是 一 个 杰 
) 6 


1. 下 漳 造 型 与 模板 / 通用 性 


为 了 使 这 些 集合 能 够 重复 使 用 ， 或 者 “再 生 ”，Java 提 供 了 一 种 通用 类 
型 ， 以 前 曾 把 它 叫 作 “Object*。 单 根 结构 意味 看 、 所 有 东西 归根 结 底 痢 
是 一 个 对 象 "! 所 以 容纳 了 Object 的 一 个 集合 实际 可 以 容纳 任何 东西 。 这 
使 我 们 对 它 的 重复 使 用 变 得 非常 简便 。 


为 使 用 这 样 的 一 个 集合 ， 只 需 添加 指 癌 它 的 对 象 句 柄 即 可 ， 以 后 可 以 通 
过 句柄 重新 使 用 对 象 。 但 由 于 集合 只 能 容纳 Object， 上 所 以 在 我 们 同 集 合 
里 添加 对 象 句柄 时 ， 它 会 上 济 造 型 成 Object， 这 样 便 丢 失 了 它 的 里 份 或 





















































者 标识 信息 。 再 次 使 用 它 的 时 候 ， 会 得 到 一 个 Object 句柄 ， 而 非 指 同 我 
们 早先 置 入 的 那个 类 型 的 句柄 。 所 以 怎样 才能 归还 它 的 本 来 面 够 ， 调 用 
早先 置 入 集合 的 那个 对 象 的 有 用 接口 呢 ? 


在 这 里 ， 我 们 再 次 用 到 了 造型 (Cast) 。 但 这 一 次 不 是 在 分 级 结构 中 上 

溯 造 型 成 一 种 更 “通用 ”的 类 型 。 而 是 下 漳 造 型 成 一 种 更 “特殊 ”的 类 型 。 

这 种 造型 方法 叫 作 “下 调 造 型 ” (Downcasting) 。 举 个 例子 来 说 ， 我 们 知 

THE EY ET ee, Circle Cll) 属于 Shape (几何 形状 〉 的 一 种 类 

型 ， 所 以 上 漳 造 型 是 安全 的 。 但 我 们 不 知道 一 个 Object 到 底 是 Circle 还 是 

E 除非 确切 地 知道 和 目 己 要 操 
J 是 什么 。 


但 这 也 不 是 绝对 和 危险 的 ， 因 为 假如 下 渊 造型 成 错误 的 东西 ， 会 得 到 我 们 
称 为 “违例 ”(Exception ) 的 一 种 运行 期 错误 。 我 们 稍 后 即 会 对 此 进行 解 
释 。 但 在 从 一 个 集合 提取 对 象 句 柄 时 ， 必 须 用 某 种 方式 准确 地 记 住 它们 
是 什么 ， 以 保证 下 调 造 型 的 正确 进行 。 


下 调 造 型 和 运行 期 检查 都 要 求 花 额外 的 时 间 来 运行 程序 ， 而 且 程序 员 必 
须 付 出 额外 的 精力 。 既 然 如 此 ， 我 们 能 不 能 创建 一 个 “智能 ”集合 ， 令 其 
知道 自己 容纳 的 类 型 呢 ? 这 样 做 可 消除 下 调 造 型 的 必要 以 及 潜在 的 错 
误 。 管 案 是 肯定 的 ， 我 们 可 以 采用 “参数 化 类 型 "*， 它 们 是 编译 右 能 目 动 
定制 的 类 ， 可 与 特定 的 类 型 配合 。 例 如 ， 通 过 使 用 一 个 参数 化 集合 ， 编 
译 右 可 对 那个 集合 进行 定制 ， 使 其 只 接受 Shape， 而 且 只 提取 Shape。 


参数 化 类 型 是 C++ 一 个 重要 的 组 成 部 分 ， 这 部 分 是 C++ 没有 单 根 结构 的 
缘故 。 在 C++ 中 ， 用 于 实现 参数 化 类 型 的 关键 字 是 template (PRA) 。 
Java 目 前 尚未 提供 参数 化 类 型 ， 因 为 由 于 使 用 的 是 单 根 结构 ， 上 所 以 使 用 
它 显得 有 些 笨 拙 。 但 这 并 不 能 保证 以 后 的 版 本 不 会 实现 ， 

为 “generic” 这 个 词 已 被 Java“ 保 留 到 将 来 实现 ”( 在 Ada 语 言 

中 , “generic” 被 用 来 实现 它 的 模板 ) 。Java 采 取 的 这 种 关键 字 保 留 机 制 
其 实 经 党 让 人 摸 不 着 头脑 ， 很 难 断 定 以 后 会 发 生 什 么 事情 。 

1.7.4 清除 时 的 困境 : 由 谁 负 贡 清除 ? 

每 个 对 象 都 要 求 资源 才能 “生存 ”， 其 中 最 令 人 注目 的 资源 是 内 存 。 如 果 
不 再 需要 使 用 一 个 对 象 ， 就 必须 将 其 清除 ， 以 便 释放 这 些 资源 ， 以 便 其 


他 对 象 使 用 。 如 果 要 解决 的 是 非常 简单 的 问题 ， 如 何 清除 对 象 这 个 问题 
并 不 显得 很 突出 : 我们 创建 对 象 ， 在 需要 的 时 候 调 用 它 ， 然 后 将 其 清除 
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举 个 例子 来 说 ， 假 设 我 们 要 设计 一 套 系统 ， 用 它 管理 一 个 机 场 的 空中 交 
通 ( 同 样 的 模型 也 可 能 适 于 省 理 一 个 仓库 的 货柜 、 或 者 一 套 影 带 出 租 系 
统 、 或 者 宠物 店 的 宠物 房 。 这 初 看 似乎 十 分 简单 :构造 一 个 集合 用 来 容 
纳 飞 机 ， 然 后 创建 一 架 新 飞机 ， 将 其 置 入 集合 。 对 进入 空中 交通 管制 区 
的 所 有 飞机 都 如 此 处 理 。 至 于 清除 ， 在 一 架 飞 机 离开 这 个 区 域 的 时 候 把 
它 简 单 地 删 去 即 可 。 


但 事情 并 没有 这 么 简单 ， 可 能 还 需要 为 一 僚 系 统 来 记录 与 飞机 有 关 的 数 
据 。 当 然 ， 和 控制 井 的 主要 功能 不 同 ， 这 些 数据 的 重要 性 可 能 一 开始 并 
不 显露 出 来 。 例 如 ， 这 条 记录 反映 的 可 能 是 离开 机 场 的 所 有 小 飞机 的 飞 
行 计划 。 所 以 我 们 得 到 了 由 小 飞机 组 成 的 另 一 个 集合 。 一 旦 创建 了 一 个 
飞机 对 象 ， 如 果 它 是 一 架 小 飞机 ， 那 么 也 必须 把 它 置 入 这 个 集合 。 然 后 
在 系统 空间 时 期 ， 需 对 这 个 集合 中 的 对 象 进行 一 些 后 台 处 理 。 


问题 现在 显得 更 复杂 了 : 如 何 才能 知道 什么 时 间 删 除 对 象 呢 ? 用 完 对 象 
后 ， 系 统 的 其 他 某 些 部 分 可 能 仍然 要 发 挥 作用 。 同 样 的 问题 也 会 在 其 他 
大 量 场合 出 现 ， 而 且 在 程序 设计 系统 中 (如 C++) ， 在 用 完 一 个 对 象 之 
后 必须 明确 地 将 其 删除 ， 所 以 问题 会 变 得 异常 复杂 GRO) 


©: 注意 这 一 点 只 对 内 存 堆 里 创建 的 对 象 成 并 (用 new 命 令 创 建 的 )。 
但 在 另 一 方面 ， 对 这 儿 描 述 的 问题 以 及 其 他 所 有 和 负 见 的 编程 问题 来 说 ， 
都 要 求 对 象 在 内 存 堆 里 创建 。 


在 Java 中 ， 垃 圾 收集 器 在 设计 时 已 考 夸 到 了 内 存 的 释放 问题 〈 尽 管 这 并 
不 包括 清除 一 个 对 象 涉 及 到 的 其 他 方面 ) 。 垃 圾 收集 器 “知道 ”一 个 对 象 
在 什么 时 候 不 再 使 用 ， 然 后 会 目 动 释放 那个 对 象 占据 的 内 存 空 间 。 采 用 
这 种 方式 ， 另 外 加 上 所 有 对 象 都 从 单个 根 类 Object 继承 的 事实 ， 而 且 由 
于 我 们 只 能 在 内 存 堆 中 以 一 种 方式 创建 对 象 ， 所 以 Java 的 编程 要 比 
et 0 a 
I 大 量 障 但。 


1. 垃圾 收集 器 对 效率 及 灵活 性 的 影响 


既然 这 是 如 此 好 的 一 种 手段 ， 为 什么 在 C++ 里 没有 得 到 充分 的 发 挥 呢 ? 
我 们 当然 要 为 这 种 编程 的 方便 性 付出 一 定 的 代价 ， 代 价 融 是 运行 期 的 开 

















销 。 正 如 早先 提 到 的 那样 ， 在 C++ 中 ， 我 们 可 在 堆栈 中 创建 对 象 。 在 这 
种 情况 下 ， 对 象 会 得 以 目 动 清除 〈 但 不 具有 在 运行 期 间 随 心 所 欲 创建 对 
象 的 灵活 性 ) 。 在 堆栈 中 创建 对 象 是 为 对 象 分 配 存储 空间 最 有 效 的 一 种 
方式 ， 也 是 释放 那些 空间 最 有 效 的 一 种 方式 。 在 内 存 堆 (Heap〉 中 创建 
对 象 可 能 要 付出 昂 贯 得 多 的 代价 。 如 果 总 是 从 同一 个 基础 类 继承 ， 并 使 
所 有 函数 调用 部 具有 “ 同 质 多 形 ” 特 征 ， 那 么 也 不 可 避免 地 需要 付出 一 定 
的 代价 。 但 垃圾 收集 器 是 一 种 特殊 的 问题 ， 因 为 我 们 永远 不 能 确定 它 什 
么 时 候 局 动 或 者 要 人 花 多 长 的 时 间 。 这 意味 着 在 Java 程 序 执 行 期 间 ， 存 在 
者 一 种 不 连 员 的 因素 。 所 以 在 杂 些 特殊 的 场合 ， 我 们 必须 避免 用 它 一 一 
比如 在 一 个 程序 的 执行 必须 保持 稳定 、 连 贯 的 时 候 (通常 把 它们 叫 
E ERRE 
ED) 。 


O: 根据 本 书 一 些 技术 性 读者 的 反馈 ， 有 一 个 现成 的 实时 Java 系 统 
(www.newmonics.com) 确实 能 够 保证 垃圾 收集 器 的 效能 。 


C++ 语言 的 设计 者 曾经 癌 C 程 序 员 发 出 请 求 〈 而 且 做 得 非 营 成功) ， 不 
要 希望 在 可 以 使 用 C 的 任何 地 方 ， 同 语言 里 加 入 可 能 对 C++ 的 速度 或 使 
用 造成 影 啊 的 任何 特性 。 这 个 目的 达到 了 ， 但 代价 就 是 C++ 的 编程 不 可 
避免 地 复杂 起 来 。Java 比 C++ 简 单 ， 但 付出 的 代价 是 效率 以 及 一 定 程度 
ee 
































1.8 违例 控制 : 解雇 错误 


从 最 古老 的 程序 设计 语言 开始 ， 错 误 控 制 一 直 都 是 设计 者 们 需要 解决 的 
一 个 大 问题 。 由 于 很 难 设 计 出 一 套 完美 的 错误 控制 方案 ， 许 多 语言 干脆 
将 问题 简单 地 忽略 反 ， 将 其 转 尹 给 库 设 计 人 员 。 对 大 多 数 错 误 控 制 方案 
来 说 ， 最 主要 的 一 个 问题 是 它们 严重 依赖 程序 员 的 警觉 性 ， 而 不 是 依赖 
语言 本 里 的 强制 标准 。 如 果 程 厅 员 不 够 警惕 一 一 大 比较 匆忙 ， 这 几乎 是 
肯定 会 发 生 的 一 一 程序 所 依赖 的 错误 控制 方案 便 会 失效 。 


“违例 控制 ?将 错误 控制 方案 内 置 到 程序 设计 语言 中 ， 有 时 甚 全 内 建 到 操 
作 系 统 内 。 这 里 的 “违例 ”(Exception〉 属于 一 个 特殊 的 对 象 ， 它 会 从 产 
生 错 误 的 地 方 “ 扔 ?或 “ 掷 ? 出 来 。 随 后 ， 这 个 违例 会 被 设计 用 于 控制 特定 
类 型 错误 的 “违例 控制 占 * 捕 获 。 在 情况 变 得 不 对 劲 的 时 候 ， 可 能 有 几 个 
违例 控制 器 并 行 捕获 对 应 的 违例 对 象 。 由 于 采用 的 是 独立 的 执行 路 径 ， 
所 以 不 会 干扰 我 们 的 常规 执行 代码 。 这 样 便 使 代码 的 编写 变 得 更 加 简 

单 ， 因 为 不 必 经 常 性 强制 检查 代码 。 除 此 以 外 ,“ 掷 ?出 的 一 个 违例 不 同 
于 从 函数 返回 的 错误 值 ， 也 不 同 于 由 函数 设置 的 一 个 标志 。 那 些 错 误 值 
或 标志 的 作用 是 指示 一 个 错误 状态 ， 是 可 以 忽略 的 。 但 违例 不 能 被 包 

略 ， 所 以 肯定 能 在 人 菜 个 地 方 得 到 处 置 。 最 后 ， 利 用 违例 能 够 可 徘 地 从 一 
个 糟糕 的 环境 中 恢复 。 此 时 一 般 不 需要 退出 ， 我 们 可 以 采取 茶 些 处 理 ， 
恢复 程序 的 正常 执行 。 显 然 ， 这 样 编制 出 来 的 程序 显得 更 加 可 靠 。 


Java 的 违例 控制 机 制 与 大 多 数 程序 设计 语言 都 有 所 不 同 。 因 为 在 Java 
中 ， 违 例 控 制 模 块 是 从 一 开始 束 封 闭 好 的 ， 所 以 必须 使 用 它 ! 如 果 没 有 
目 己 写 一 些 代码 来 正确 地 控制 违例 ， 就 会 得 到 一 条 编译 期 出 错 提示 。 这 
样 可 保证 程序 的 连贯 性 ， 使 错误 控制 变 得 更 加 容易 。 


注意 违例 控制 并 不 属于 一 种 面 癌 对 象 的 特性 ， 尽 管 在 面 问 对 象 的 程序 设 
计 语 言 中 ， 违 例 通 常 是 用 一 个 对 象 表 示 的 。 早 在 面向 对 象 语言 问世 以 
前 ， 违 例 控 制 就 已 经 存在 了 。 




















1.9 多 线程 


在 计算 机 编程 中 ， 一 个 基本 的 概念 就 是 同时 对 多 个 任务 加 以 控制 。 许 多 
程序 设计 问题 都 要 求 程 序 能 够 俘 下 手头 的 工作 ， 改 为 处 理 其 他 一 些 问 

题 ， 再 返回 主 进程 。 可 以 通过 多 种 途径 达到 这 个 目的 。 最 开始 的 时 候 ， 
那些 拥有 机 器 低级 知识 的 程序 员 编 写 一 些 “ 中 断 服 务 例 程 >”， 主 进程 的 暂 
停 是 通过 硬件 级 的 中 断 实现 的 。 尽 管 这 是 一 种 有 用 的 方法 ， 但 纺 出 的 程 
序 很 难 移植 ， 由 此 造成 了 为 一 类 的 代价 高 郧 问 题 。 


有 些 时候 ， 中 断 对 那些 实时 性 很 强 的 任务 来 说 是 很 有 必要 的 。 但 还 存在 
其 他 许多 问题 ， 它 们 只 要 求 将 问题 划分 进入 独立 运行 的 程序 片断 中 ， 使 
整个 程序 能 更 迅速 地 啊 应 用 户 的 请 求 。 在 一 个 程序 中 ， 这 些 独 立 运行 的 
片断 叫 作 “线程 ”CThread) ， 利 用 它 编 程 的 概念 就 叫 作 * 多 线程 处 理 ”。 
多 线程 处 理 一 个 常见 的 例子 就 是 用 户 界 面 。 利 用 线程 ， 用 户 可 按 下 一 个 
按钮 ， 然 后 程序 会 立即 作出 啊 应 ， 而 不 是 让 用 户 等 竺 程序 完成 了 当前 任 
务 以 后 才 开始 啊 应 。 


最 开始 ， 线 程 只 是 用 于 分 配 单个 处 理 器 的 处 理 时 间 的 一 种 工具 。 但 假如 
操作 系统 本 刁 文 持 多 个 处 理 器 ， 那 么 每 个 线程 都 可 分 配给 一 个 不 同 的 处 
理 句 ， 真 正 进入 “并 行 运算 ”状态 。 从 程序 设计 语言 的 角度 看 ， 多 线程 操 
作 最 有 价值 的 特性 之 一 就 是 程序 员 不 必 关 心 到 确 使 用 了 多 少 个 处 理 器 。 
程序 在 馆 辑 意义 上 被 分 割 为 数 个 线程 ;假如 机 器 本 号 安 装 了 多 个 处 理 
项 ， 那 么 程序 会 运行 得 更 快 ， 毋 需 作 出 任何 特殊 的 调 校 。 


根据 前 面 的 论述 ， 大 家 可 能 感觉 线程 处 理 非常 简单 。 但 必须 注意 一 个 问 
题 : 共享 资源 ! 如 果 有 多 个 线程 同时 运行 ， 而 且 它 们 试图 访问 相同 的 资 
源 ， 残 会 过 到 一 个 问题 。 举 个 例子 来 说 ， 两 个 进程 不 能 将 信息 同时 发 送 
给 一 台 打 印 机 。 为 解决 这 个 问题 ， 对 那些 可 共 至 的 资源 来 说 (比如 打印 
机 〉， 它 们 在 使 用 期 间 必 须 进 入 锁定 状态 。 所 以 一 个 线程 可 将 资源 锁 

定 ， 在 完成 了 它 的 任务 后 ， 再 解 开 《〈 释 放 ) 这 个 锁 ， 使 其 他 线程 可 以 接 
痢 使 用 同样 的 资源 。 


Java 的 多 线程 机 制 已 内 建 到 语言 中 ， 这 使 一 个 可 能 较 复 杂 的 问题 变 得 简 
单 起 来 。 对 多 线程 处 理 的 支持 是 在 对 象 这 一 级 支持 的 ， 所 以 一 个 执行 线 
程 可 表达 为 一 个 对 象 。Java 也 提供 了 有 限 的 资源 锁定 方案 。 它 能 锁定 任 
何 对 象 占用 的 内 存 ( 内 存 实际 是 多 种 共有 至 资源 的 一 种 ) ， 所 以 同一 时 间 


























只 能 有 一 个 线程 使 用 特定 的 内 存 空 间 。 为 达到 这 个 目的 ， 需 要 使 用 
synchronized 关 键 字 。 其 他 类 型 的 资源 必须 由 程序 员 明 确 锁 定 ， 这 通常 
要 求 程 序 员 创建 一 个 对 象 ， 用 它 代表 一 把 锁 ， 所 有 线程 在 访问 那个 资源 


时 都 必须 检查 这 把 锁 。 


1.10 永久 性 


创建 一 个 对 象 后 ， 只 要 我 们 需要 ， 它 就 会 一 直 存 在 下 去 。 但 在 程序 结束 
运行 时 ， 对 象 的 “生存 期 ”也 会 宣告 结束 。 尽 管 这 一 现象 表面 上 非常 合 

理 ， 但 深入 退守 就 会 发 现 ， 假 如 在 程序 停止 运行 以 后 ， 对 象 也 能 继续 存 
在 ， 并 能 保留 它 的 全 部 信息 ， 那 么 在 茶 些 情况 下 将 是 一 件 非常 有 价值 的 
事情 。 下 次 月 动 程序 时 ， 对 象 仍然 在 那里 ， 里 面 保留 的 信息 仍然 是 程序 
上 一 次 运行 时 的 那些 信息 。 当 然 ， 可 以 将 信息 写 入 一 个 文件 或 者 数据 

库 ， 从 而 达到 相同 的 效果 。 但 尽管 可 将 所 有 东西 都 看 作 一 个 对 月 ， 如 果 
能 将 对 象 声 明成 “永久 性 ”， 并 令 其 为 我 们 照看 其 他 所 有 细节 ， 无 颖 也 是 
一 件 相当 方便 的 事情 。 


Java 1.1 提 供 了 对 “有 限 永 久 性 ?的 文 持 ， 这 意味 着 我 们 可 将 对 象 简单 地 保 
存 到 磁盘 上 ， 以 后 任何 时 间 都 可 取 回 。 之 所 以 称 它 为 "有限 ” 的 ， 是 由 于 
我 们 仍然 需要 明确 发 出 调用 ， 进 行 对 象 的 保存 和 取 回 工作 。 这 些 工 作 不 
能 目 动 进行 。 在 Java 未 来 的 版 本 中 ， 对 “永久 性 ”的 文 持 有 望 更 加 全 面 。 























1.11 Java 和 因特网 


既然 Java 不 过 另 一 种 类 型 的 程序 设计 语言 ， 大 家 可 能 会 奇怪 它 为 什么 值 
得 如 此 重视 ， 为 什么 还 有 这 么 多 的 人 认为 它 是 计算 机 程序 设计 的 一 个 里 
程 碑 呢 ? 如 果 您 来 自 一 个 传统 的 程序 设计 背景 ， 那 么 答案 在 刚 开 始 的 时 
候 并 不 是 很 明显 。Java 除 了 可 解决 传统 的 程序 设计 问题 以 外 ， 还 能 解决 
World Wide Web (万 维 网 ) 上 的 编程 问题 。 


1.11.1 什么 是 Web? 


Web 这 个 词 刚 开始 显得 有 些 泛泛 ， 似 乎 “冲浪 *、*“ 网 上 存在 ”以 及 " 主 
页 ”等 等 都 和 它 拉 上 了 一 些 关系 。 其 至 还 有 一 种 <Internet 综 合 症 ” 的 说 
法 ， 对 许多 人 狂热 的 上 网 行为 提出 了 质疑 。 我 们 在 这 里 有 必要 作 一 些 深 
入 的 探讨 ， 但 在 这 之 前 ， 必 须 理解 客户 机 / 服务 器 系统 的 概念 ， 这 是 充 
斥 着 许多 令 人 迷惑 的 问题 的 又 一 个 计算 领域 。 


1. 客户 机 / Rait 


客户 机 / 服务 器 系统 的 基本 思想 是 我 们 能 在 一 个 统一 的 地 方 集中 存放 信 
恩 资 源 。 一 般 将 数据 集中 保存 在 茶 个 数据 库 中 ， 根 据 其 他 人 或 者 机 器 的 
请 求 将 信息 投递 给 对 方 。 客 户 机 / 服务 器 概述 的 一 个 关键 在 于 信息 

是 “集中 存放 ”的 。 所 以 我 们 能 方便 地 更 改 信 息 ， 然 后 将 修改 过 的 信息 发 
放 给 信息 的 消费 者 。 将 各 种 元 系 集 中 到 一 起 ， 信 息 仓 库 、 用 于 投递 信息 
的 软件 以 及 信息 及 软件 所 在 的 那 台 机 器 ， 它 们 联合 起 来 便 叫 作 “ 服 务 
ae” (Server) 。 而 对 那些 驻 留 在 远程 机 器 上 的 软件 ， 它 们 需要 与 服务 器 
通信 ， 取 回信 息 ， 进 行 适当 的 处 理 ， 然 后 在 远程 机 器 上 显示 出 来 ， 这 些 
BLA ESA” (Client) 。 


这 样 看 来 ， 客 户 机 / 服务 器 的 基本 概念 并 不 复杂 。 这 里 要 注意 的 一 个 主 
要 问题 是 单个 服务 器 需要 同时 问 多 个 客户 提供 服务 。 在 这 一 机 制 中 ， 通 
常 少 不 了 一 套数 据 库 管理 系统 ， 使 设计 人 员 能 将 数据 布局 封装 到 表格 

中 ， 以 获得 最 优 的 使 用 。 除 此 以 外 ， 系 统 经 第 允许 客户 将 新 信息 插入 一 
个 服务 器 。 这 意味 着 必须 确保 客户 的 新 数据 不 会 与 其 他 客户 的 新 数据 冲 
突 ， 或 者 说 需要 保证 那些 数据 在 加 入 数据 库 的 时 候 不 会 丢失 《〈 用 数据 库 
的 术语 来 说 ， 这 叫 作 “事务 处 理 *”) 。 客 户 软件 发 生 了 改变 之 后 ， 它 们 必 
须 在 客户 机 器 上 构建 、 调 试 以 及 安装 。 所 有 这 些 会 使 问题 变 得 比 我 们 一 























般 想 象 的 复杂 得 多 。 为 外 ， 对 多 种 类 型 的 计算 机 和 操作 系统 的 文 持 也 是 
一 个 大 问题 。 最 后 ， 性 能 的 问题 显得 尤为 重要 : 可 能 会 有 数 百 个 客户 同 
时 问 服 务 器 发 出 请 求 。 所 以 任何 微小 的 延误 都 是 不 能 忽视 的 。 为 太 可 能 
缓解 潜伏 的 问题 ， 程 序 员 需 要 谨慎 地 分 散 任务 的 处 理 负 担 。 一 般 可 以 考 
虑 让 客户 机 负担 部 分 处 理 任务 ， 但 有 时 亦 可 分 派 给 服务 器 所 在 地 的 其 他 
机 器 ， 那 些 机 器 亦 叫 作 * 中 间 件 ”《“ 中 间 件 也 用 于 改进 对 系统 的 维护 ) 。 


所 以 在 具体 实现 的 时 候 ， 其 他 人 发 布 信息 这 样 一 个 简单 的 概念 可 能 变 得 
异常 复杂 。 有 时 甚至 会 使 人 产生 完全 无 从 着 手 的 感觉 。 客 户 机 / 服务 器 
的 概念 在 这 时 就 可 以 大 显 身 手 了 。 事 实 上 ， 大 约 有 一 半 的 程序 设计 活动 
都 可 以 采用 客户 机 / 服务 器 的 结构 。 这 种 系统 可 负责 从 处 理 订 单 及 信用 
卡 交 易 ， 一 直到 发 布 各 类 数据 的 方方面面 的 任务 一 一 股票 市 场 、 科 学 研 
完 、 政 府 运作 等 等 。 在 过 去 ， 我 们 一 般 为 单独 的 问题 采取 单独 的 解决 方 
案 ; 每 次 都 要 设计 一 套 新 方案 。 这 些 方 案 无 论 创 建 还 是 使 用 都 比较 困 
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2. Webxe ~ TEKIN ARS AF 


Web 实 际 就 是 一 套 规模 巨大 的 客户 机 / 服务 器 系统 。 但 它 的 情况 要 复杂 
一 些 ， 因 为 所 有 服务 堪 和 客户 都 同时 存在 于 单个 网 络 上 面 。 但 我 们 没 必 
要 了 解 更 进一步 的 细节 ， 因 为 唯一 要 关心 的 就 是 一 次 建立 同一 个 服务 器 
i EA 
F) o 


最 开始 的 时 候 ， 这 是 一 个 简单 的 单 向 操作 过 程 。 我 们 同一 个 服务 器 发 出 
请 求 ， 它 同 我 们 回 传 一 个 文件 ， 由 于 本 机 的 浏览 器 软件 ( 亦 即 “ 客 

户 ?或 “客户 程序 ”") 负责 解释 和 格式 化 ， 并 在 我 们 面前 的 屏幕 上 正确 地 
显示 出 来 。 但 人 们 不 和 久 就 不 满足 于 只 从 一 个 服务 器 传递 网 页 。 他 们 希望 
获得 完全 的 客户 机 / 服务 器 能 力 ， 使 客户 《程序 ) 也 能 反馈 一 些 信息 到 
服务 器 。 比 如 而 望 对 服务 器 上 的 数据 库 进 行 检索 ， 回 服务 咒语 加 新 信 
恩 ， 或 者 下 一 份 订单 等 等 (这 也 提供 了 比 以 前 的 系统 更 高 的 安全 要 
pa Caen 0 eter tegen 


Webi ims NARA TIO [REN a: 某 个 信息 可 在 任何 类 型 的 计 
算 机 上 显示 出 来 ， 毋 需 任 何 改动 。 然 而 ， 浏 览 吉 仍然 显得 很 原始 ， 在 用 
户 迅速 增多 的 要 求 面前 显得 有 些 力不从心 。 和 它们 的 交互 能 力 不 够 强 ， 而 















































且 对 服务 器 和 因特网 都 造成 了 一 定 程度 的 和 干扰。 这 是 由 于 每 次 采取 一 些 
要 求 纺 程 的 操作 时 ， 必 须 将 信息 反馈 回 服务 器 ， 在 服务 器 那 一 瑞 进 行 处 
理 。 所 以 完全 可 能 需要 等 待 数秒 乃至 数 分 钟 的 时 间 才 会 发 现 自 己 刚 才 拼 
音 了 一 个 单词 。 由 于 浏览 器 只 是 一 个 纯粹 的 查看 程序 ， 所 以 连 最 简单 的 
计算 任务 都 不 能 进行 〈 当 然 在 另 一 方面 ， 它 也 显得 非常 安全 ， 因 为 不 能 
在 本 机 上 面 执行 任何 程序 ， 避 开 了 程序 错误 或 者 病毒 的 又 扰 ) 。 


为 解决 这 个 问题 ， 人 们 采取 了 许多 不 同 的 方法 。 最 开始 的 时 候 ， 人 们 对 
图 形 标准 进行 了 改进 ， 使 浏览 器 能 显示 更 好 的 动画 和 视频 。 为 解决 剩 下 
的 问题 ， 唯 一 的 办 法 就 是 在 客户 端 〈 浏 览 句 ) 内 运行 程序 。 这 就 叫 
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1.11.2 客户 端 编程 (注释 @)) 


Web 最 初 及 用 的 “服务 絮 一 浏览 器 ” 方 采 可 提供 交互 式 内 容 ， 但 这 种 交互 
能 力 完 全 由 服务 器 提供 ， 为 服务 器 和 因特网 融 来 了 不 小 的 负担 。 服 务 器 
一 般 为 客户 浏览 锅 产 生 毅 态 网 页 ， 由 后 者 简单 地 解释 并 显示 出 来 。 基 本 
HTML 语 言 提供 了 简单 的 数据 收集 机 制 ， 文字 输入 框 、 复 选 枉 、 单 选 

钮 、 列 表 以 及 下 拉 列 表 等 ， 另 外 还 有 一 个 按钮 ， 只 能 由 程序 规定 重新 设 
置 表单 中 的 数据 ， 以 便 回 传 给 服务 器 。 用 户 提交 的 信息 通过 所 有 Web 服 
务 器 均 能 支持 的 “通用 网 关 接 口 ”(CGI) 回 传 到 服务 器 。 包 含 在 提交 数 
扬中 的 文字 指示 CGI 该 如 何 操 作 。 最 第 见 的 行动 是 运行 位 于 服务 占 的 一 
个 程序 。 那 个 程序 一 般 保存 在 一 个 名 为 “cgi-bin” 的 目录 中 〈 按 下 Web 页 
内 的 一 个 按钮 时 ， 请 注意 一 下 浏览 旨 顶 部 的 地 址 窗 ， 经 常 都 能 发 现 “cgi- 
bin" HF) 。 大 多 数 语 言 都 可 用 来 编制 这 些 程序 ， 但 其 中 最 利 见 的 是 
Perl。 这 是 由 于 Per 是 专 为 文字 的 处 理 及 解释 而 设计 的 ， 所 以 能 在 任何 
服务 器 上 安装 和 使 用 ， 无 论 采 用 的 处 理 韦 或 操作 系统 是 什么 。 


©: 本 节 内 容 改 编 自 某 位 作者 的 一 篇 文章 。 那 篇 文章 最 早出 现在 位 于 
www.mainspring.com 的 Mainspring 上 。 本 节 的 采用 已 征 得 了 对 方 的 同 
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今天 的 许多 Web 站 点 都 严格 地 建立 在 CGI 的 基础 上 ， 事 实 上 几乎 所 有 事 
情 都 可 用 CGI 做 到 。 唯 一 的 问题 就 是 啊 应 时 间 。CGI 程 序 的 啊 应 取决 于 
需要 传送 多 少数 据 ， 以 及 服务 器 和 因特网 两 方面 的 负担 有 多 重 〈 而 且 
CGI 程 序 的 局 动 比较 慢 ) 。Web 的 早期 设计 者 并 未 预料 到 当初 绎 综 有 余 
的 带宽 很 快 就 变 得 不 够 用 ， 这 正 是 大 量 应 用 充斥 网 上 造成 的 结案 。 例 




















如 ， 此 时 任何 形式 的 动态 图 形 显示 都 几乎 不 能 连贯 地 显示 ， 因 为 此 时 必 
须 创 建 一 个 GIF 文件 ， 再 将 图 形 的 每 种 变化 从 服务 器 传递 给 客户 。 而 且 
大 家 应 该 对 输入 表单 上 的 数据 校 验 有 着 深刻 的 体会 。 原 来 的 方法 是 我 们 
按 下 网 页 上 的 提交 按钮 (Submit) ; 数据 回 传 给 服务 器 ;服务 器 启动 一 
个 CGI 程序 ， 检 查 用 户 输入 是 否 有 错 ; 格式 化 一 个 HTML 页 ， 通 知 可 能 
遇 到 的 错误 ， 并 将 这 个 页 回 传 给 我 们 ， 随 后 必须 回 到 原先 那个 表单 页 ， 
再 输入 一 遍 。 这 种 方法 不 仅 速 度 非 常 慢 ， 也 显得 非常 繁琐 。 


解决 的 办 法 就 是 客户 端的 程序 设计 。 运 行 Web 浏 览 器 的 大 多 数 机 器 都 拥 
有 足够 强 的 能 力 ， 可 进行 其 他 大 量 工 作 。 与 此 同时 ， 原 始 的 静态 HTML 
方法 仍然 可 以 采用 ， 它 会 一 直 等 到 服务 器 送 回 下 一 个 页 。 客 户 站 编程 意 
| 
(互动 ) 能 


对 客户 并 编程 的 讨论 与 常规 编程 问题 的 讨论 并 没有 太 大 的 区 别 。 采 用 的 
参数 肯定 是 相同 的 ， 只 是 运行 的 平台 不 同 : Web 浏 览 占 就 象 一 个 有 限 的 
操作 系统 。 无 论 如 何 ， 我 们 仍然 需要 编程 ， 仍 然 会 在 客 尸 并 编程 中 过 到 
大 量 问题 ， 同 时 也 有 很 多 解决 的 方案 。 在 本 节 剩 下 的 部 分 里 ， 我 们 将 对 
这 些 问 题 进行 一 番 概 括 ， 并 介绍 在 客户 站 编程 中 采取 的 对 策 。 


1. 插件 


明和 客户 端 编 程 迈 进 的 时 候 ， 最 重要 的 一 个 问题 就 是 插件 的 设计 。 利 用 插 
件 ， 程 序 员 可 以 方便 地 为 浏览 圳 语 加 新 功能 ， 用 户 只 需 下 载 一 些 代 码 ， 
把 它们 “插入 ”浏览 器 的 适当 位 置 即 可 。 这 些 代 码 的 作用 是 告诉 浏览 

器 “从 现在 开始 ， 你 可 以 进行 这 些 新 活动 了 ”《〈 仅 需 下 载 这 些 插入 一 
次 ) 。 有 些 快 速 和 功能 强大 的 行为 是 通过 插件 潜 加 到 浏览 融 的 。 但 插件 
的 编写 并 不 是 一 件 简单 的 任务 。 在 我 们 构建 一 个 特定 的 站 点 时 ， 可 能 3 
不 布 望 涉 及 这 方面 的 工作 。 对 客户 端 程序 设计 来 说 ， 搬 件 的 价值 在 于 它 
允许 专业 程序 员 设 计 出 一 种 新 的 语言 ， 并 将 那 种 语言 添加 到 浏览 器 ， 同 
时 不 必 经 过 浏览 右 原 创 者 的 许可 。 由 此 可 以 看 出 ， 插 件 实 际 是 浏览 器 的 
一 个 “后 门 ”， 人 允许 创建 新 的 客户 端 程序 设计 语言 《尽管 并 非 所 有 语言 都 
是 作为 插件 实现 的 )。 


2. 脚本 编制 语言 


插件 造成 了 脚本 编制 语言 的 焊 炸 性 增长 。 通 过 这 种 脚本 语言 ， 可 将 用 于 
自己 客户 端 程序 的 源码 直接 插入 HTML 页 ， 而 对 那 种 语言 进行 解释 的 插 
































件 会 在 HTML 页 显示 的 时 候 目 动 激活 。 脚 本 语言 一 般 都 倾 问 于 尽量 简 
化 ， 易 于 理解 。 而 且 由 于 它们 是 从 属于 HTML 页 的 一 些 简单 正文 ， 所 以 
只 需 回 服务 器 发 出 对 那个 页 的 一 次 请 求 ， 即 可 非常 快 地 载 入 。 缺 点 是 我 
们 的 代码 全 部 暴露 在 人 们 面前 。 男 一 方面 ， 由 于 通常 不 用 脚本 编制 语言 
做 过 份 复杂 的 事情 ， 所 以 这 个 问题 暂且 可 以 放 在 一 边 。 


脚本 语言 真正 面 癌 的 是 特定 类 型 问题 的 解决 ， 其 中 主要 涉及 如 何 创建 更 
丰富、 更 具有 互动 能 力 的 图 形 用 尸 界面 GUI) 。 然 而 ， 脚 本 语言 也 许 
能 解决 客户 端 编程 中 80% 的 问题 。 你 碰 到 的 问题 可 能 完全 就 在 那 80% 里 
面 。 而 且 由 于 脚本 编制 语言 的 宗旨 是 尽 可 能 地 简化 与 快速 ， 所 以 在 考虑 
其 他 更 复杂 的 方案 之 前 (如 Java 及 ActiveX) ， 首 先 应 想 一 下 脚本 语言 是 
伸 可 行 。 


目前 讨论 得 最 多 的 脚本 编制 语言 包括 JavaScript( 它 与 Java 没 有 任何 关 

Aa; 之 所 以 叫 那 个 名 字 ， 完 全 是 一 种 市 场 策略 ) 、VBScript〈 同 Visual 
Basic 很 相似 ) 以 及 TcVyTKk 〈 来 源 于 流行 的 路 平台 GUI 构造 语言 ) 。 当 然 
还 有 其 他 许多 语言 ， 也 有 许多 正在 开发 中 。 


JavaScript 也 许 是 目 常 用 的 ， 它 得 到 的 支持 也 最 全 面 。 无 论 
NetscapeNavigator, Microsoft Internet Explorer， 还 是 Opera， 目 前 都 提供 
了 对 JavaScript 的 文 持 。 除 此 以 外 ， 市 面 上 讲述 JavaScript 的 书籍 也 要 比 
讲述 其 他 语言 的 书 多 得 多 。 有 些 工 具 还 能 利用 JavaScript 目 动产 生 网 页 。 
当然 ， 如 果 你 已 经 有 Visual Basic 或 者 TclyTkK 的 深厚 功底 ， 当 然 用 它们 要 
简单 得 多 ， 起 人 码 可 以 避免 学 习 新 语言 的 烦恼 (解决 Web 方 面 的 问题 就 已 
经 够 让 人 头痛 了 ) 。 























3. Java 


如 果 说 一 种 脚本 编制 语言 能 解决 80%% 的 客户 端 程 序 设 计 问 题 ， 那 么 剩 下 
的 20% 叉 该 怎么 办 呢 ? 它们 属于 一 些 高 难度 的 问题 吗 ? 目前 最 流行 的 方 
案 就 是 Java。 它 不 仅 是 一 种 功能 强大 、 高 度 安全 、 可 以 跨 平 台 使 用 以 及 
国际 通用 的 程序 设计 语言 ， 也 是 一 种 具有 旺盛 生命 力 的 语言 。 对 Java 的 
扩展 是 不 断 进行 的 ， 提 供 的 语言 特性 和 库 能 够 很 好 地 解决 传统 语言 不 能 
解决 的 问题 ， 比 如 多 线程 操作 、 数 据 库 访问 、 连 网 程序 设计 以 及 分 布 式 
计算 等 等 。Java 通 过 “程序 片 ”(Applet) 巧妙 地 解决 了 客户 端 编程 的 问 
题 。 


EFA (或 “小 应 用 程序 ”) 是 一 种 非常 小 的 程序 ， 只 能 在 Web 浏 览 器 中 





运行 。 作 为 Web 页 的 一 部 分 ， 程 序 片 代码 会 目 动 下 载 回 来 〈 这 和 网 页 中 
的 图 片 差不多 ) 。 激 活 程 序 片 后 ， 它 会 执行 一 个 程序 。 程 序 片 的 一 个 优 
扩 体 现在 ;通过 程序 片 ， 一 旦 用 户 需 要 客户 软件 ， 软 件 就 可 从 服务 器 目 
动 下 载 回 来 。 它 们 能 自动 取得 客户 软件 的 最 新 版 本 ， 不 会 出 错 ， 也 没有 
重新 安 沪 的 胀 烦 。 由 于 Java 的 设计 原理 ， 程 序 员 只 雷 要 创建 程序 的 一 个 
版 本 ， 那 个 程序 能 在 几乎 所 有 计算 机 以 及 安装 了 Java 解 释 占 的 浏览 占 中 
运行 。 由 于 Java 是 一 种 全 功能 的 编程 语言 ， 所 以 在 癌 服 务 器 及 出 一 个 请 
求 之 前 ， 我 们 能 先 在 客户 端 做 完 尽 可 能 多 的 工作 。 例 如 ， 再 也 不 必 通 过 
因特网 传送 一 个 请 求 表单 ， 再 由 服务 器 确定 其 中 是 否 存在 一 个 拼写 或 者 
其 他 参数 错误 。 大 多 数 数据 校 验 工 作 均 可 在 客户 端 完成 ， 没 有 必要 坐 在 
计算 机 前 面 焦急 地 等 竺 服务 器 的 啊 应 。 这 样 一 来 ， 不 仅 速 度 和 啊 应 的 灵 
人 敏 度 得 到 了 极 大 的 提高 ， 对 网 络 和 服务 器 造成 的 负担 也 可 以 明显 减轻 ， 
这 对 保障 因特网 的 畅通 是 至 关 重 要 的 。 


与 脚本 程序 相 比 ，Java 程 序 片 的 男 一 个 优点 是 它 采 用 编译 好 的 形式 ， 所 
以 客户 端 看 不 到 源码 。 当 然 在 另 一 方面 ， 反 编译 Java 程 序 片 也 并 不 是 件 
难事 ， 而 且 代 码 的 隐藏 一 般 并 不 是 个 重要 的 问题 。 大 家 要 注意 另外 两 个 
重要 的 问题 。 正 如 本 书 以 前 会 讲 到 的 那样 ， 编 译 好 的 Java 程 序 片 可 能 包 
含 了 许多 模块 ， 所 以 要 多 次 “命中 ” (访问 ) 服务 器 以 便 下 载 〈 在 Java 1.1 
中 ， 这 个 问题 得 到 了 有 效 的 改善 利用 Java 压 缩 档 ， 即 JAR 文 件 
它 人 允许 设计 者 将 所 有 必要 的 模块 都 封装 到 一 起 ， 供 用 户 统 一 下 载 )。 在 
男 一 方面 ， 脚 本 程序 是 作为 Web 页 正文 的 一 部 分 集成 到 Web 页 内 的 。 这 
种 程序 一 般 都 非常 小 ， 可 有 效 减 少 对 服务 器 的 点 击 数 。 男 一 个 因素 是 学 
习 方 面 的 问题 。 不 管 你 平时 听 别 人 怎么 说 ，Java 都 不 是 一 种 十 分 容易 便 
可 学 会 的 语言 。 如 果 你 以 前 是 一 名 Visual Basic 程 序 员 ， 那 么 转 辐 
VBScript 会 是 一 种 最 快捷 的 方案 。 由 于 VBScript 可 以 解决 大 多 数 典 型 的 
客户 机 / 服务 器 问题 ， 所 以 一 旦 上 手 ， 束 很 难 下 定 决 心 再 去 学 习 Java。 

如 果 对 脚本 编制 语言 比较 熟 ， 那 么 在 转 同 Java 之 前 ， 建 议 先 熟 悉 一 下 

JavaScript 或 者 VBScript， 因 为 它们 可 能 已 经 能 够 满足 你 的 需要 ， 不 必 经 
历 学 习 Java 的 艰苦 过 程 。 


4. ActiveX 


在 菏 种 程度 上 ，Java 的 一 个 有 力 苋 争 对 手 应 该 是 微软 的 ActiveX， 尺 管 它 
采用 的 是 完全 不 同 的 一 套 实现 机 制 。ActiveX 最 早 是 一 种 纯 Windows 的 
方案 。 经 过 一 家 独立 的 专业 协会 的 努力 ，ActiveX 现 在 已 具备 了 路 平台 
使 用 的 能 力 。 实 际 上 ，ActiveX 的 意思 是 “假如 你 的 程序 同 它 的 工作 环境 

































































正常 连接 ， 它 就 能 进入 Web 页 ， 并 在 文 持 ActiveX 的 浏览 器 中 运行 ”〈IE 
固化 了 对 ActiveX 的 支持 ， 而 Netscape 需 要 一 个 插件 ) 。 所 以 ，ActiveX 
并 没有 限制 我 们 使 用 一 种 特定 的 语言 。 比 如 ， 假 设 我 们 已 经 是 一 名 有 经 
验 的 Windows 程 序 员 ， 能 熟练 地 使 用 象 C++、Visual Basic 或 者 
BorlandDelphi 那 样 的 语言 ， 束 能 几乎 不 加 任何 学 习 地 创建 出 ActiveX 组 
件 。 事实 上 ，ActiveX 是 在 我 们 的 Web 页 中 使 用 “历史 遗留 ”代码 的 最 佳 途 
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自动 下 载 和 通过 因特网 运行 程序 听 起 来 就 象 是 一 个 病毒 制造 者 的 梦想 。 
在 客户 端的 编程 中 ，ActiveX 带 来 了 最 让 人 头痛 的 安全 问题 。 点 击 一 个 
Web 站 点 的 时 候 ， 可 能 会 随同 HTML 网 页 传 回 任何 数量 的 东西 : GIF 文 
件 、 脚 本 代码 、 编 译 好 的 Java 代 码 以 及 ActiveX 组 件 。 有 些 是 无 害 的 ; 
GIF 文件 不 会 对 我 们 造成 任何 危害 ， 而 脚本 编制 语言 通常 在 自己 可 做 的 
事情 上 有 着 很 大 的 限制 。Java 也 设计 成 在 一 个 安全 “ 沙 箱 ?里 在 它 的 程序 
片 中 运行 ， 这 样 可 防止 操作 位 于 沙 箱 以 外 的 磁盘 或 者 内 存 区 域 。 


ActiveX 是 所 有 这 些 里 面 最 让 人 担心 的 。 用 ActiveX 编 写 程序 就 象 编制 
Windows 应 用 程序 可 以 做 上 自己 想 做 的 任何 事情 。 下 载 回 一 个 
ActiveX 组 件 后 ， 它 完全 可 能 对 我 们 磁盘 上 的 文件 造成 破坏 。 当 然 ， 对 
那些 下 载 回 来 并 不 限于 在 Web 浏 览 器 内 部 运行 的 程序 ， 它 们 同样 也 可 能 
破坏 我 们 的 系统 。 从 BBS 下 载 回 来 的 病毒 一 直 是 个 大 问题 ， 但 因特网 的 
速度 使 得 这 个 问题 变 得 更 加 复杂 。 


目前 解决 的 办 法 是 “数字 签名 ”， 代 码 会 得 到 权威 机 构 的 验证 ， 显 示 出 它 
的 作者 是 谁 。 这 一 机 制 的 基础 是 认为 病毒 之 所 以 会 传播 ， 是 由 于 它 的 编 
制 者 匿名 的 缘故 。 所 以 假如 去 挥 了 匿名 的 因素 ， 所 有 设计 者 部 不 得 不 为 
它们 的 行为 负责 。 这 似乎 是 一 个 很 好 的 主意 ， 因 为 它 使 程序 显得 更 加 正 
规 。 但 我 对 它 能 消除 恶意 因素 持 怀疑 态度 ， 因 为 假如 一 个 程序 便 含 有 
Bug， 那 么 同样 会 造成 问题 。 


Java 通 过 “ 沙 箱 ” 来 防止 这 些 问 题 的 发 生 。Java 解 释 器 内 骨 于 我 们 本 地 的 
Web 浏 览 占 中 ， 在 程序 片 装 载 时 会 检查 所 有 有 嫌疑 的 指令 。 特 别 地 ， 程 
厅 片 根本 没有 权力 将 文件 写 进 磁盘 ， 或 者 删除 文件 (这 是 病毒 最 喜欢 做 
的 事情 之 一 ) 。 我 们 通 冲 认为 程序 片 是 安全 的 。 而 且 由 于 安全 对 于 营建 
一 套 可 徘 的 客户 机 / 服务 器 系统 至 关 重 要 ， 所 以 会 给 病毒 留 下 漏洞 的 所 
有 错误 都 能 很 快 得 到 修复 〈 浏 览 右 软件 实际 需要 强行 齐 守 这 些 安 全 规 





















































则 ， 而 有 些 浏 览 器 则 人 允许 我 们 选择 不 同 的 安全 级 别 ， 防 止 对 系统 不 同 程 
度 的 访问 ) 。 


大 家 或 许 会 怀疑 这 种 限制 是 否 会 妨碍 我 们 将 文件 写 到 本 地 和 磁盘。 比如， 
我 们 有 时 需要 构建 一 个 本 地 数据 库 ， 或 将 数据 保存 下 来 ， 以 便 日 后 离线 
使 用 。 最 早 的 版 本 似乎 每 个 人 都 能 在 线 做 任何 敏感 的 事情 ， 但 这 很 快 就 
变 得 非常 不 现实 《尽管 低 价 “ 互 联网 工具 ”有 一 天 可 能 会 满足 大 多 数 用 户 
的 需要 ) 。 解 诀 的 方案 是 “ 签 了 名 的 程序 片 *， 它 用 公共 密 钥 加 密 算 法 验 
证 程序 片 确实 来 自 它 所 声称 的 地 方 。 当 然 在 通过 验证 后 ， 签 了 名 的 一 个 
程序 片 仍然 可 以 开始 清除 你 的 磁盘 。 但 从 理论 上 说 ， 既 然 现 在 能 够 找到 
创建 人 “ 算 帐 ”， 他 们 一 般 不 会 干 这 种 等 事 。Java 1.1 为 数字 签名 提供 了 一 
个 框架 ， 在 必要 时 ， 可 让 一 个 程序 片 “ 走 ”到 沙 箱 的 外 面 来 。 


数字 签名 遗漏 了 一 个 重要 的 问题 ， 那 就 是 人 们 在 因特网 上 移动 的 速度 。 
如 下 载 回 一 个 错误 百出 的 程序 ， 而 它 很 不 过 地 真 的 干 了 茶 些 臣 事 ， 需 要 
多 和 久 的 时 间 才 能 发 党 这 一 点 昵 ? 这 也 许 是 几 天 ， 也 可 能 几 周 之 后 。 发 现 
了 之 后 ， 又 如 何 奶 踩 当初 掌 事 的 程序 呢 《〈 以 及 它 当 时 的 黄 任 有 多 大 ) ? 


6. 因特网 和 内 联网 


Web 是 解决 客户 机 / 服务 器 问题 的 一 种 常用 方案 ， 所 以 最 好 能 用 相同 的 
技术 解雇 此 类 问题 的 一 些 * 子 集 ”， 特 别 是 公司 内 部 的 传统 客户 机 / 服务 
器 问题 。 对 于 传统 的 客户 机 / 服务 器 模式 ， 我 们 面临 的 问题 是 拥有 多 种 
不 同类 型 的 客户 计算 机 ， 而 且 很 难 安装 新 的 客户 软件 。 但 通过 Web 浏 览 
器 和 客户 端 编程 ， 这 两 类 问题 都 可 得 到 很 好 的 解决 。 知 一 个 信息 网 络 局 
限于 一 家 特定 的 公司 ， 那 么 在 将 Web 技 术 应 用 于 它 之 后 ， 即 可 称 其 

为 “内 联网 ”(Intranet〉， 以 示 与 国际 性 的 “因特网 ”(Internet) AB. A 
联网 提供 了 比 因特网 更 大 的 安全 级 别 ， 因 为 可 以 物理 性 地 控制 对 公司 内 
部 服务 器 的 使 用 。 说 到 培训 ， 一 般 只 要 人 们 理解 了 浏览 器 的 常规 概念 ， 
就 可 以 非常 轻松 地 掌握 网 页 和 程序 片 之 间 的 差异 ， 所 以 学 习 新 型 系统 的 
开销 会 大 幅度 减少 。 


安全 问题 将 我 们 引入 客户 端 编程 领域 一 个 似乎 是 自动 形成 的 分 支 。 寿 程 
序 是 在 因特网 上 运行 ， 由 于 无 从 知晓 它 会 在 什么 平台 上 运行 ， 所 以 编程 
时 要 特别 留意 ， 防 范 可 能 出 现 的 编程 错误 。 需 作 一 些 跨 平 台 处 理 ， 以 及 
适当 的 安全 防范 ， 比 如 采用 某 种 脚本 语言 或 者 Java。 


但 假如 在 内 联网 中 运行 ， 面 临 的 一 些 制约 因素 就 会 发 生变 化 。 全 部 机 器 









































均 为 Intel/Windows 平 台 是 件 很 平常 的 事情 。 在 内 联网 中 ， 需 要 对 自己 代 
码 的 质量 负责 。 而 且 一 旦 发 现 错误 ， 就 可 以 马上 改正 。 除 此 以 外 ， 可 能 
己 经 有 了 一 些 “ 历 史 遗 留 * 的 代码 ， 并 用 较 传 统 的 客户 机 / 服务 器 方式 使 
用 那些 代码 。 但 在 进行 升级 时 ， 每 次 都 要 物理 性 地 安装 一 道 客 户 程 序 。 
滔 费 在 升级 安装 上 的 时 间 是 转移 到 浏览 右 的 一 项 重要 原因 。 使 用 了 浏览 
ava, FRM DUS, MAES SISA A METH. WR 
真 的 是 罕 涉 到 这 样 的 一 个 内 联网 中 ， 最 明智 的 方法 是 采用 ActiveX， 而 
非 试 图 采用 一 种 新 的 语言 来 改写 程序 代码 。 


面临 客户 端 编程 问题 令 人 困惑 的 一 系列 解决 方案 时 ， 最 好 的 方 采 是 先 做 
一 次 投资 / 回报 分 析 。 请 总 结 出 问题 的 全 部 制约 因素 ， 以 及 什么 才 是 最 
快 的 方案 。 由 于 客户 端 程序 设计 仍然 要 编程 ， 所 以 无 论 如 何 部 该 针对 上 自 
己 的 特定 情况 采取 最 好 的 开发 途径 。 这 是 准备 面 对 程 序 开 发 中 一 些 不 可 
避免 的 问题 时 ， 我 们 可 以 作出 的 最 佳 姿态 。 


1.11.3 服务 器 端 编程 


我 们 的 整个 讨论 都 忽略 了 服务 器 端 编程 的 问题 。 如 果 辐 服务 堪 发 出 一 个 
请 求 ， 会 发 生 什 么 事情 ? 大 多 数 时候 的 请 求 都 是 很 简单 的 一 个 “把 这 个 
文件 发 给 我 ?。 浏 览 器 随后 会 按 适当 的 形式 解释 这 个 文件 ， 作 为 HTML 
页 、 一 幅 图 、 一 个 Java 程 序 片 、 一 个 脚本 程序 等 等 。 同 服务 器 发 出 的 较 
复杂 的 请 求 通常 涉及 到 对 一 个 数据 库 进 行 操作 (事务 处 理 ) 。 其 中 最 党 
见 的 就 是 发 出 一 个 数据 库 检 索 命 令 ， 得 到 结果 后 ， 服 务 器 会 把 它 格 式 化 
成 HTML 页 ， 并 作为 结果 传 回 来 〈 当 然 ， 假 如 客户 通过 Java 或 者 某 种 脚 
本 语言 具有 了 更 高 的 智能 ， 那 么 原始 数据 就 能 在 客户 端 发 送 和 格式 化 ; 
这 样 做 速度 可 以 更 快 ， 也 能 减轻 服务 器 的 负担 ) 。 另 外 ， 有 时 需要 在 数 
据 库 中 注册 上 自己 的 名 字 【比如 加 入 一 个 组 时 ) ， 或 者 回 服 务 器 发 出 一 份 
订单 ， 这 如 涉及 到 对 那个 数据 库 的 修改 。 这 类 服务 器 请 求 必须 通过 服务 
器 端的 一 些 代码 进行 ， 我 们 称 其 为 “服务 器 端的 编程 >。 在 传统 意义 上 ， 
服务 器 端 编程 是 用 Perl 和 CGI 脚本 进行 的 ， 但 更 复杂 的 系统 已 经 出 现 。 

其 中 包括 基于 Java 的 Web 服 务 器 ， 它 允许 我 们 用 Java 进 行 所 有 服务 器 端 
Ste, 5 ANREP SM EARS PEP” (Servlet) 。 


1.11.4 一 个 独立 的 领域 : 应 用 程序 
与 Java 有 关 的 大 多 数 争论 都 是 与 程序 片 有 关 的 。Java 实 际 是 一 种 常规 用 


途 的 程序 设计 语言 ， 可 解决 任何 类 型 的 问题 ， 至 少 理论 上 如 此 。 而 且 正 
如 前 面 指出 的 ， 可 以 用 更 有 效 的 方式 来 解决 大 多 数 客户 机 / 服务 需 问 
































题 。 如 果 将 视线 从 程序 片 届 上 转 开 《同时 放宽 一 些 限 制 ， 比 如 禁止 写 盘 
等 ) ， 就 进入 了 第 规 用 途 的 应 用 程序 的 广阔 领域 。 这 种 应 用 程序 可 独立 
运行 ， 毋 需 浏 览 器 ， 就 象 普 通 的 执行 程序 那样 。 在 这 儿 ，Java 的 特色 并 
不 仅仅 反应 在 它 的 移植 能 力 ， 也 反映 在 编程 本 身上 。 融 象 贯 罕 全 书 都 会 
讲 到 的 那样 ，Java 提 供 了 许多 有 用 的 特性 ， 使 我 们 能 在 较 短 的 时 间 里 创 
建 出 比 用 从 前 的 程序 设计 语言 更 健壮 的 程序 。 


但 要 注意 任何 东西 都 不 是 十 全 十 美的 ， 我 们 为 此 也 要 付出 一 些 代 价 。 其 
中 最 明显 的 是 执行 速度 放 慢 了 《尽管 可 对 此 进行 多 方面 的 调整 ) IEE 
何 语言 一 样 ，Java 本 吴 也 存在 一 些 限 制 ， 使 得 它 不 十 分 适合 解决 东 些 特 
殊 的 编程 问题 。 但 不 管 怎 样 ，Java 都 是 一 种 正在 快速 友 展 的 语言 。 随 着 
el a ee neceaeicee es etertoe ge 




















1.12 分 析 和 设计 


面 问 对 象 的 范式 是 思考 程序 设计 时 一 种 新 的 、 而 且 全 然 不 同 的 方式 ， 许 
多 人 最 开始 都 会 在 如 何 构 造 一 个 项 目 上 皱 起 了 眉 尖 。 事 实 上 ， 我 们 可 以 
作出 一 个 “好 ?的 设计 ， 它 能 充分 利用 OOP 提 供 的 所 有 优点 。 


有 关 OOP 分 析 与 设计 的 书籍 大 多 数 都 不 尽 如 人 意 。 其 中 的 大 多 数 书 都 充 
斥 着 莫名其妙 的 话语 、 举 拙 的 笔调 以 及 许多 听 起 来 似乎 很 重要 的 声明 

QER) 。 我 认为 这 种 书 最 好 压缩 到 一 章 左 右 的 空间 ， 至 多 写成 一 本 
非常 注 的 书 。 具 有 讽刺 意味 的 是 ， 那 些 特别 专注 于 复杂 事物 管理 的 人 往 
往 在 写 一 些 浅显 、 明 白 的 书 上 面 大 费 周 革 ! 如 朵 不能 说 得 简单 和 直接 ， 
一 定 没 多 少 人 喜欢 看 这 方面 的 内 容 。 毕 竟 ，OOP 的 全 部 宗旨 就 是 让 软件 
开发 的 过 程 变 得 更 加 容易 。 尽 管 这 可 能 影响 了 那些 喜欢 解决 复杂 问题 的 
人 的 生计 ， 但 为 什么 不 从 一 开始 就 把 事情 弄 得 简单 些 呢 ? 因此 ， 和 希望 我 
能 从 开始 就 为 大 家 打下 一 个 恨 好 的 基础 ， 尽 可 能 用 几 个 段落 来 说 清楚 分 
析 与 设计 的 问题 。 


©: 最 好 的 入 门 书 仍 然 是 Grady Booch 的 《Object-Oriented Design 
withApplications， 第 2 版 本 》，Wiely & Sons 于 1996 年 出 版 。 这 本 书 讲 得 
很 有 深度 ， 而 且 通 俗 易 介 ， 尽 管 他 的 记号 方法 对 大 多 数 设 计 来 说 都 显得 
不 必要 地 复杂 。 


1.12.1 不 要 迷失 


在 整个 开发 过 程 中 ， 最 重要 的 事情 就 是 : 不 要 将 目 己 迷失 ! 但 事实 上 这 
种 事情 很 容易 及 生 。 大 多 数 方法 都 设计 用 来 解决 最 大 范围 内 的 问题 。 当 
然 ， 也 存在 一 些 特别 困难 的 项 目 ， 需 要 作者 付出 更 为 艰辛 的 努力 ， 或 者 
付出 更 大 的 代价 。 但 是 ， 大 多 数 项 目 都 是 比较 “ 闻 规 > 的， 所 以 一 般 都 能 
作出 成 功 的 分 析 与 设计 ， 而 且 只 需 用 到 推荐 的 一 小 部 分 方法 。 但 无 论 多 
么 有 限 ， 某 些 形式 的 处 理 总 是 有 益 的 ， 这 可 使 整个 项 目的 开发 更 加 容 

易 ， 总 比 直 接 了 当 开 始 编码 好 ! 


也 惑 是 说 ， 假 如 你 正在 考察 一 种 特殊 的 方法 ， 其 中 包含 了 大 量 细节 ， 并 
推荐 了 许多 步骤 和 文档 ， 那 么 仍然 很 难 正 确 判断 目 己 该 在 何 时 停止 。 时 
刻 提 醒目 己 注意 以 下 几 个 问题 : 























(1) 对 象 是 什么 ?” 《怎样 将 自己 的 项 目 分 割 成 一 系列 单独 的 组 件 ? ) 
(2) 它们 的 接口 是 什么 ”《〈 需 要 将 什么 消 妃 发 给 每 一 个 对 象 ? ) 


在 确定 了 对 象 和 它们 的 接口 后 ， 便 可 着 手 编写 一 个 程序 。 出 于 对 多 方面 
原因 的 考虑 ， 可 能 还 需要 比 这 更 多 的 说 明 及 文档 ， 但 要 求 掌 握 的 资料 绝 
对 不 能 比 这 还 少 。 


整个 过 程 可 划分 为 四 个 阶段 ， 阶 段 0 刚刚 开始 采用 某 些 形式 的 结构 。 
1.12.2 阶段 0: 拟 出 一 个 计划 


第 一 步 是 决定 在 后 面 的 过 程 中 采取 哪些 步骤 。 这 听 起 来 似乎 很 简单 〈《 事 
实 上 ， 我 们 这 儿 资 的 一 切 都 似乎 很 简单 ) ， 但 很 常见 的 一 种 情况 是 : 有 
些 人 甚至 没有 进入 阶段 1， 便 忙 忙 居民 地 开始 编写 代码 。 如 果 你 的 计划 
本 来 束 是 “直接 开始 开始 编码 "那样 做 当然 也 无 可 非议 〔 寿 对 自己 要 解 
决 的 问题 已 有 很 透彻 的 理解 ， 便 可 考虑 那样 做 )。 但 最 低 程 度 也 应 同意 
目 己 该 有 个 计划 。 


在 这 个 阶段 ， 可 能 要 决定 一 些 必要 的 附加 处 理 络 构 。 但 非常 不 笠 ， 有 些 
程序 员 写 程序 时 喜欢 随心 所 欲 ， 他 们 认为 “该 完成 的 时 候 目 然 会 完成 。 
这 样 做 刚 开 始 可 能 不 会 有 什么 问题 ， 但 我 党 得 假如 能 在 整个 过 程 中 设置 
儿 个 标志 ， 或者“ 路标?"， 将 更 有 益 于 你 集中 注意 力 。 这 恐怕 比 单纯 地 为 
了 “完成 工作 ”而 工作 好 得 多 。 人 至 少 ， 在 达到 了 一 个 义 一 个 的 目标 ， 经 过 
了 一 个 接 一 个 的 路 标 以 后 ， 可 对 目 己 的 进度 有 清晰 的 把 握 ， 干 劲 也 会 相 
应 地 提高 ， 不 会 产生 “路 遥 漫 漫 无 期 ”的 感觉 。 


座 我 刚 开 始 学 习 故 事 结 构 起 〈 我 想 有 一 天 能 写本 小 说 出 来 ) ， 就 一 直 坚 
持 这 种 做 法 ， 感 沉 就 象 简单 地 让 文字 “ 流 ? 到 纸 上 。 在 我 写 与 计算 机 有 关 
的 东西 时 ， 友 现 结构 要 比 小 说 简单 得 多 ， 所 以 不 需要 考虑 太 多 这 方面 的 
问题 。 但 我 仍然 制订 了 整个 写作 的 结构 ， 使 自己 对 要 写 什么 做 到 心中 有 
数 。 因 此 ， 即 使 你 的 计划 就 是 直接 开始 写 程序 ， 仍 然 需 要 经 历 以 下 的 阶 
段 ， 同 时 向 目 己 提出 一 些 特定 的 问题 。 


1.12.3 阶段 1: 要 制作 什么 ? 


在 上 一 代 程 序 设计 中 即 “ 过 程 化 或 程序 化 设计 ”) ， 这 个 阶段 称 为 “ 建 
并 需求 分 析 和 系统 规格 "。 当 然 ， 那 些 操 作 今天 已 经 不 再 需要 了 ， 或 者 









































至 少 改 换 了 形式 。 大 量 令 人 头痛 的 文档 资料 已 成 为 历史 。 但 当时 的 初 袁 
和 是 好 的 。 需 求 分 析 的 意思 是 “建立 一 系列 规则 ， 根 据 它 判断 任务 什么 时 
候 完 成 ， 以 及 客户 怎样 才能 满意 ”。 系 统 规格 则 表示 “这 里 是 一 些 具体 的 
说 明 ， 让 你 知 着 程序 需要 做 什么 《而 不 是 怎样 做 ) 才能 满足 要 求 ”。 需 
求 分 析 实 际 就 是 你 和 客户 之 间 的 一 份 合 约 〈 即 使 客户 融 在 本 公司 内 部 工 
作 ， 或 者 是 其 他 对 象 及 系统 ) 。 系 统 规格 是 对 所 面临 问题 的 最 高 级 别 的 
一 种 揭示 ， 我 们 依据 它 判 断 任务 是 否 完成 ， 以 及 需要 花 多 长 的 时 间 。 由 
于 这 些 都 需要 取得 参与 者 的 一 致 同意 ， 所 以 我 建议 尽 可 能 地 简化 它们 

一 一 最 好 采用 列表 和 基本 图 表 的 形式 一 一 以 市 省 时 间 。 可 能 还 会 面临 田 
一 些 限制 ， 需 要 把 它们 扩充 成 为 更 大 的 文档 。 


我 们 特别 要 注意 将 重点 放 在 这 一 阶段 的 核心 问题 上 ， 不 要 纠缠 于 细 枝 末 
Wo MAD te: 决定 及 用 什么 系统 。 对 这 个 问题 ， 最 有 价值 的 
工具 就 是 一 个 名 为 “使 用 条 件 ” 的 集合 。 对 那些 采用 “假如 .….…， 系统 该 
怎样 做 ? "形式 的 问题 ， 这 便 是 最 有 说 服 力 的 回答 。 例 如 , “假如 客户 需 
要 提取 一 张 现金 文 紧 ， 但 当时 又 没有 这 么 多 的 现金 储备 ， 那 么 自动 取 殖 
机 该 怎样 反应 ? ”对 这 个 问题 ,，“ 使 用 条 件 ” 可 以 指示 自动 取 球 机 在 那 
种 条件” 下 的 正确 操作 。 


应 尽 可 能 总 结 出 目 己 系统 的 一 套 完 整 的 “使 用 条 件 ? 或 者 “应 用 场合 "。 一 
旦 完成 这 个 工作 ， 就 相当 于 摸 清 了 想 让 系统 完成 的 核心 任务 。 由 于 将 重 
扩 放 在 “使 用 条 件 ”* 上 ， 一 个 很 好 的 效果 就 是 它们 总 能 让 你 放 精 力 放 在 最 
关键 的 东西 上 ， 并 防止 自己 分 心 于 对 完成 任务 关系 不 大 的 其 他 事情 上 

面 。 也 就 是 说 ， 只 要 掌握 了 一 套 完整 的 “使 用 条 件 ”， 就 可 以 对 自己 的 系 
统 作 出 清晰 的 描述 ， 并 转移 到 下 一 个 阶段 。 在 这 一 阶段 ， 也 有 可 能 无 法 
完全 和 擎 握 系 统 日 后 的 各 种 应 用 场合 ， 但 这 也 没有 关系 。 只 要 肯 花 时 间 ， 

所 有 问题 都 会 目 然而 然 暴 露出 来 。 不 要 过 份 在 意 系 统 规格 的 “完美 ”， 人 否 
则 也 容易 产生 挫败 感 和 焦煤 情绪 。 


在 这 一 阶段 ， 最 好 用 几 个 简单 的 段落 对 自己 的 系统 作出 描述 ， 然 后 围 经 
它们 再 进行 扩充 ， 添 加 一 些 “ 名 词 * 和 “动词 "。“ 名 词 * 目 然 成 为 对 象 ， 

而 “动词 "自然 成 为 要 整合 到 对 象 接口 中 的 “方法 "。 只 要 亲自 试 着 做 一 
做 ， 就 会 发 现 这 是 多 么 有 用 的 一 个 工具 ; 有 些 时 候 ， 它 能 帮助 你 完成 绝 
大 多 数 的 工作 。 


尽管 仍 处 在 初级 阶段 ， 但 这 时 的 一 些 日 程 安排 也 可 能 会 非常 管用 。 我 们 
现在 对 目 己 要 构建 的 东西 应 该 有 了 一 个 较 全 面 的 认识 ， 所 以 可 能 已 经 感 
沉 到 了 它 大 概 会 花 多 长 的 时 间 来 完成 。 此 时 要 考虑 多 方面 的 因素 : 如 采 












































估计 出 一 个 较 长 的 日 程 ， 那 么 公司 也 许 决 定 不 再 继续 下 去 ; 或 者 一 名 主 
管 己 经 估算 出 了 这 个 项 目 要 花 多 长 的 时 间 ， 并 会 试看 影响 你 的 估计 。 但 
无 论 如 何 ， 最 好 从 一 开始 就 草拟 出 一 份 “ 诚 实 ” 的 时 间 表 ， 以 后 再 进行 一 
些 暂 时 难以 作出 的 决策 。 目 前 有 许多 技术 可 帮助 我 们 计算 出 准确 的 日 程 
安排 ( 束 象 那些 预测 股票 市 场 起 落 的 技术 〉 ， 但 通常 最 好 的 方法 还 是 依 
赖 自 己 的 经 验 和 直觉 〈 不 要 忘记 ， 直 觉 也 要 建立 在 经 验 上 ) 。 感 觉 一 下 
大 概 需 要 花 多 长 的 时 间 ， 然 后 将 这 个 时 间 加 倍 ， 再 加 上 10%。 你 的 感觉 
可 能 是 正确 的 ;“ 也 许 ” 能 在 那个 时 间 里 完成 。 但 “加 倍 ” 使 那个 时 间 更 加 
充裕 ,，“10%” 的 时 间 则 用 于 进行 最 后 的 推 设 和 深化 。 但 同时 也 要 对 此 癌 
上 级 主管 作出 适当 的 解释 ， 无 论 对 方 有 什么 抱 仿 和 修改 ， 只 要 明确 地 告 
诉 他 们 : 这 样 的 一 个 日 程 安排 ， 只 是 我 的 一 个 估计 ! 


1.12.4 阶段 2， 如何 构建 ? 


在 这 一 阶段 ， 必 须 拿 出 一 套 设计 方案 ， 并 解释 其 中 包含 的 各 类 对 象 在 外 
观 上 是 什么 样子 ， 以 及 相互 间 是 如 何 沟 通 的 。 此 时 可 考虑 采用 一 种 特殊 
的 图 表 工 具 :“ 统 一 建 模 语言 ”(UML) 。 请 到 http://www.rational.com 去 
下 载 一 份 UML 规 格 书 。 作 为 第 1 阶段 中 的 描述 工具 ，UML 也 是 很 有 帮助 
的 。 此 外 ， 还 可 用 它 在 第 2 阶段 中 处 理 一 些 图 表 《〈 如 流程 图 ) 。 当 然 并 
非 一 定 要 使 用 UML， 但 它 对 你 会 很 有 帮助 ， 特 别 是 在 希望 描绘 一 张 详 
尽 的 图 表 ， 让 许多 人 在 一 起 研究 的 时 候 。 除 UML 外 ， 还 可 选择 对 对 象 
以 及 它们 的 接口 进行 文字 化 摘 述 《就 象 我 在 《Thinking in C++》 里 说 的 
那样 ， 但 这 种 方法 非常 原始 ， 发 挥 的 作用 亦 较 有 限 。 


我 兽 有 一 次 非常 成 功 的 咨询 经 历 ， 那 时 涉及 到 一 小 组 人 的 初始 设计 。 他 
们 以 前 还 没有 构建 过 OOP《〈 面 癌 对 象 程序 设计 ) 项 目 ， 将 对 象 国 在 白板 
上 面 。 我 们 谈 到 各 对 象 相互 间 该 如 何 沟通 〈 通 信 ) ， 并 删除 了 其 中 的 一 
部 分 ， 以 及 蔡 换 了 另 一 部 分 对 象 。 这 个 小 组 〈 他 们 知道 这 个 项 目的 目的 
是 什么 〉 实 际 上 已 经 制订 出 了 设计 方 膝 ; 他 们 自己 “拥有 ”了 设计 ， 而 不 
古 让 设计 自然 而 然 地 显露 出 来 。 我 在 那里 做 的 事情 就 是 对 设计 进行 指 

吐 ， 提 出 一 些 适 当 的 问题 ， 尝 试 作出 一 些 假设 ， 并 从 小 组 中 得 到 反馈 ， 
以 便 修 改 那些 假设 。 这 个 过 程 中 最 美妙 的 事情 就 是 整个 小 组 并 不 是 通过 
学 习 一 些 抽 象 的 例子 来 进行 面向 对 象 的 设计 ， 而 是 通过 实践 一 个 真正 的 
设计 来 掌握 OOP 的 等 门 ， 而 那个 设计 正 是 他 们 当时 手 上 的 工作 ! 


作出 了 对 对 象 以 及 它们 的 接口 的 说 明 后 ， 就 完成 了 第 2 阶段 的 工作 。 当 
然 ， 这 些 工作 可 能 并 不 完全 。 有 些 工 作 可 能 要 等 到 进入 阶段 3 才能 得 
知 。 但 这 已 经 足够 了 。 我 们 真正 需要 关心 的 是 最 终 找 出 所 有 的 对 象 。 能 





























早 些 发 现 当 然 好 ， 但 OOP 提 供 了 足够 完美 的 结构 ， 以 后 再 找 出 它们 也 不 


IRo 
1.12.5 阶段 3: 开始 创建 


读 这 本 书 的 可 能 是 程序 员 ， 现 在 进入 的 正 是 你 可 能 最 感 兴趣 的 阶段 。 由 
村 手头 上 有 一 个 计划 一 一 无 论 它 有 多 么 简要 ， 而 且 在 正式 编码 前 掌握 了 
正确 的 设计 结构 ， 所 以 会 发 现 接 下 去 的 工作 比 一 开始 就 埋头 写 程序 要 简 
单 得 多 。 而 这 正 是 我 们 想 达 到 的 目的 。 让 代码 做 到 我 们 想 做 的 事情 ， 这 
征 所 有 程序 项 目 最 终 的 目标 。 但 切 不 要 和 急 功 骨 进 ， 人 否则 只 有 得 不 偿 失 。 
根据 我 的 经 验 ， 最 后 先 拿 出 一 套 较为 全 面 的 方案 ， 使 其 尽 可 能 设想 周 

全 ， 能 满足 尽 可 能 多 的 要 求 。 给 我 的 感觉 ， 编 程 更 象 一 门 艺 术 ， 不 能 只 
是 作为 技术 活 来 看 待 。 所 有 付出 最 终 都 会 得 到 回报 。 作 为 真正 的 程序 

员 ， 这 并 非 可 有 可 无 的 一 种 素质 。 全 面 的 思考 、 周 密 的 准备 、 民 好 的 构 
造 不 仅 使 程序 更 易 构 建 与 调试 ， 也 使 其 更 易 理 解 和 维护 ， 而 那 正 是 一 套 
软件 顾 利 的 必要 条 件 。 


构建 好 系统 ， 并 令 其 运行 起 来 后 ， 必 须 进 行 实际 检验 ， 以 前 做 的 那些 需 
求 分 析 和 系统 规格 便 可 派 上 用 场 了 。 全 面 地 考察 自己 的 程序 ， 确 定 提 出 
的 所 有 要 求 均 已 满足 。 现 在 一 切 似乎 都 该 结束 了 ? 是 吗 ? 


1.12.6 阶段 4: 校订 


事实 上 ， 整 个 开发 周期 还 没有 结束 ， 现 在 进入 的 是 传统 意义 上 称 为 “ 维 
护 ” 的 一 个 阶段 。“ 维 护 ”是 一 个 比较 暧昧 的 称呼 ， 可 用 它 表 示 从 “保持 它 
按 设想 的 轨道 运行 “加 入 客户 从 前 起 了 声明 的 功能 ”或 者 更 传统 的 “ 除 
掉 暴 露出 来 的 一 切身 虫 * 等 等 意思 。 所 以 大 家 对 “维护 ”这 个 词 产生 了 许 
多 误解 ， 有 的 人 认为 : 凡是 需要 “维护 ”的 东西 ， 必 定 不 是 好 的 ， 或 者 是 
有 缺陷 的 ! 因为 这 个 词 说 明 你 实际 构建 的 是 一 个 非常 “原始 ”的 程序 ， 以 
后 需要 频 每 地 作出 改动 、 添 加 新 的 代码 或 者 防止 它 的 落后 、 退 化 等 。 因 
此 ， 我 们 需要 用 一 个 更 合理 的 词语 来 称呼 以 后 需要 继续 的 工作 。 


这 个 词 便 是 “校订 ?。 换 言 之 , “你 第 一 次 做 的 东西 并 不 完善 ， 所 以 需 为 
目 己 留 下 一 个 深入 学 习 、 认 知 的 空间 ， 再 回 过 头 去 作 一 些 改变 ”。 对 于 
要 解决 的 问题 ， 随 着 对 它 的 学 习 和 了 解 愈加 深入 ， 可 能 需要 作出 大 量 改 
动 。 进 行 这 些 工 作 的 一 个 动力 是 随 着 不 断 的 改革 优化 ， 终 于 能 够 从 目 己 
的 努力 中 得 到 回报 ， 无 论 这 需要 经 历 一 个 较 短 还 是 较 长 的 时 期 。 




































































什么 时 候 才 叫 “ 达 到 理想 的 状态 ” 呢 ? 这 并 不 仅仅 意味 着 程 序 必 须 按 要 求 
的 那样 工作 ， 并 能 适应 各 种 指定 的 “使 用 条 件 ”， 它 也 意味 着 代码 的 内 部 
结构 应 当 尺 善 尽 美 。 人 至少， 我 们 应 能 感觉 出 整个 结构 都 能 民 好 地 协调 运 
作 。 没 有 笨拙 的 语法 ， 没 有 腑 肿 的 对 象 ， 也 没有 一 些 华而不实 的 东西 。 
除 此 以 外 ， 必 须 保证 程序 结构 有 很 强 的 生命 力 。 由 于 多 方面 的 原因 ， 以 
后 对 程序 的 改动 是 必 不 可 少 。 但 必须 确定 改动 能 够 方便 和 清楚 地 进行 。 
这 里 没有 人 花 巧 可 言 。 不 仅 需 要 理解 目 己 构 建 的 是 什么 ， 也 要 理解 程序 如 
何不 断 地 进化 。 幸 运 的 是 ， 面 癌 对 象 的 程序 设计 语言 特别 适合 进行 这 类 
连续 作出 的 修改 一 一 由 对 象 建立 起 来 的 边界 可 有 效 你 证 结构 的 整体 性 ， 
并 能 防范 对 无 天 对 象 进行 的 无 谓 干 扰 、 破 坏 。 也 可 以 对 自己 的 程序 作 一 
些 看 似 激烈 的 大 变动 ， 同 时 不 会 破坏 程序 的 整体 性 ， 不 会 波及 到 其 他 代 
码 。 事 实 上 ， 对 "校订 ?的 文 持 是 OOP 非 和 重要 的 一 个 特点 。 


通过 校订 ， 可 创建 出 至 少 接近 自己 设想 的 东西 。 然 后 从 整体 上 观察 自己 
的 作品 ， 把 它 与 自己 的 要 求 比较 ， 看 看 还 短缺 什么 。 然 后 就 可 以 从 容 地 
回 过 头 去 ， 对 程序 中 不 恰当 的 部 分 进行 重新 设计 和 重新 实现 (注释 
W) 。 在 最 终 得 到 一 套 恰 当 的 方案 之 前 ， 可 能 需要 解决 一 些 不 能 回避 的 
问题 ， 或 者 至 少 解决 问题 的 一 个 方面 。 而 且 一 般 要 多 “校订 ” 几 次 才 行 
eel mene gs Meee 
ij 第 16 章 ) 。 


构建 一 套 系统 时 , “校订 ?几乎 是 不 可 避免 的 。 我 们 需要 不 断 地 对 比 目 己 
的 需求 ， 了 解 系统 是 否 目 己 实际 所 需要 的 。 有 时 只 有 实际 看 到 系统 ， 才 
能 意识 到 上 自己 需要 解决 一 个 不 同 的 问题 。 寿 认为 这 种 形式 的 校订 必然 会 
发 生 ， 那 么 最 好 尽快 拿 出 自己 的 第 一 个 版 本 ， 检 查 它 是 否 自己 希望 的 ， 
使 目 己 的 思想 不 断 趋 癌 成 熟 。 


反复 的 “校订 ? 同 “ 递 增 开 及” 有 关 密 不 可 分 的 关系 。 递 增 开 发 意味 痢 先 从 
系统 的 核心 入 手 ， 将 其 作为 一 个 框 染 实现 ， 以 后 要 在 这 个 框 染 的 基础 上 
逐渐 建立 起 系统 剩余 的 部 分 。 随 后 ， 将 准备 提供 的 各 种 功能 特性 ) 一 
个 接 一 个 地 加 入 其 中 。 这 里 最 考验 技巧 的 是 架设 起 一 个 能 方便 扩充 所 有 
目标 特性 的 一 个 框 染 (对 这 个 问题 ， 大 家 可 参考 第 16 章 的 论述 ) 。 这 样 
做 的 好 处 在 于 一 旦 令 核心 框 漆 运作 起 来 ， 要 加 入 的 每 一 项 特性 就 象 它 目 
号 内 的 一 个 小 项 目 ， 而 非 大 项 目的 一 部 分 。 此 外 ， 开 发 或 维护 阶段 合成 
的 新 特性 可 以 更 方便 地 加 入 。OOP 之 所 以 提供 了 对 递增 开发 的 文 持 ， 是 
人 
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O: 这 有 点 类 似 “ 快 速 造 型 ”。 此 时 应 着 眼 于 建立 一 个 简单 、 明 了 的 版 
本 ， 使 目 己 能 对 系统 有 个 清楚 的 把 握 。 再 把 这 个 原型 扔 挤 ， 并 正式 地 构 
建 一 个 。 快 速 造 型 最 麻烦 的 一 种 情况 就 是 人 们 不 将 原型 扔 反 ， 而 古 直 接 
在 它 的 基础 上 建造 。 如 果 再 加 上 程序 化 设计 中 “结构 ?的 缺乏 ， 就 会 导致 
一 个 混乱 的 系统 ， 致 使 维护 成 本 增加 。 


1.12.7 计划 的 回报 


如 果 没 有 仔细 拟定 的 设计 图 ， 当 然 不 可 能 建 起 一 所 房子 。 如 建立 的 是 一 
所 狗 舍 ， 尺 管 设计 图 可 以 不 必 那 么 详尽 ， 但 仍然 需要 一 些 草 图 ， 以 做 到 
心中 有 数 。 软 件 开发 则 完全 不 同 ， 它 的 “设计 图 ”《〈 计 划 ) 必须 详尽 而 完 
备 。 在 很 长 的 一 段 时 间 里 ， 和 人们 在 他 们 的 开发 过 程 中 并 没有 太 多 的 结 

构 ， 但 那些 大 型 项 目 很 容易 束 会 遭 尾 失败 。 通 过 不 断 的 摸索 ， 人 们 掌握 
了 数量 众多 的 结构 和 详细 资料 。 但 它们 的 使 用 却 使 人 提心吊胆 在 意 
似乎 需要 把 目 己 的 大 多 数 时 间 花 在 编写 文 要 上， 而 没有 多 少时 间 来 编程 
(经 常 如 此 ) 。 我 而 望 这 里 为 大 家 讲述 的 一 切 能 提供 一 条 折衷 的 道路 。 
需要 来 取 一 种 最 适合 自己 需要 (以 及 习惯 ) 的 方法 。 不 管制 订 出 的 计划 
有 多 么 小 ， 但 与 完全 没有 计划 相 比 ， 一 些 形式 的 计划 会 极 大 改善 你 的 项 
目 。 请 记 住 : 根据 估计 ， 没 有 计划 的 50% 以 上 的 项 目 都 会 失败 ! 

















1.13 Java 还 是 C++? 


Java 特 别 象 C++; 由 此 很 自然 地 会 得 出 一 个 结论 : C++ 似乎 会 被 Java 取 
代 。 但 我 对 这 个 逻辑 存 有 一 些 疑 问 。 无 论 如 何 ，C++ 仍 有 一 些 特 性 是 
Java 没 有 的 。 而 且 尽 管 已 有 大 量 保证 ， 声 称 Java 有 一 天 会 达到 或 超过 
C++ 的 速度 。 但 这 个 突破 迄今 仍 未 实现 〈 尽 管 Java 的 速度 确实 在 稳步 提 
高 ， 但 仍 未 达到 C++ 的 速度 ) 。 此 外 ， 许 多 领域 都 存在 为 数 众 多 的 
C++ 爱好 者 ， 所 以 我 并 不 认为 那 种 语言 很 快 就 会 被 另 一 种 语言 奉 代 【〈 爱 
好 者 的 力量 是 容 急 视 的 。 比 如 在 我 主持 的 一 次 “中 / 高 级 Java 研 讨 
Z” E, Allen Holub 声 称 两 种 最 常用 的 语言 是 Rexx 和 COBOL) 。 


我 感觉 Java 强 大 之 处 反映 在 与 C++ 稍 有 不 同 的 领域 。C++ 是 一 种 绝对 不 
会 试图 迎合 某 个 模子 的 语言 。 特 别 是 它 的 形式 可 以 变化 多 端 ， 以 解决 不 
同类 型 的 问题 。 这 主要 反映 在 象 Microsoft Visual C++ 和 Borland C++ 
Builder 〈 我 最 喜欢 这 个 ) 那样 的 工具 身上 。 它 们 将 库 、 组 件 模型 以 及 代 
码 生 成 工具 等 合成 到 一 起 ， 以 开发 视窗 化 的 末端 用 户 应 用 《〈 用 于 
Microsoft Windows 操 作 系 统 ) 。 但 在 另 一 方面 ，Windows 开 发 人 员 最 常 
用 的 是 什么 呢 ? 是 微软 的 Visual Basic (VB) 。 当 然 ， 我 们 在 这 儿 和 暂且 
不 提 VB 的 语法 极 易 使 人 迷惑 的 事实 即使 一 个 只 有 几 页 长 度 的 程 
序 ， 产 生 的 代码 也 十 分 难于 管理 。 从 语言 设计 的 角度 看 ， 尽 管 VB 是 那 
样 成 功 和 流行 ， 但 仍然 存在 不 少 的 缺点 。 最 好 能 够 同时 拥有 VB 那样 的 
强大 功能 和 易 用 性 ， 同 时 不 要 产生 难于 管理 的 代码 。 而 这 正 是 Java 最 吸 
引 人 的 地 方 : 作为 “下 一 代 的 VB”。 无 论 你 听 到 这 种 主张 后 有 什么 感 
觉 ， 请 无 论 如 何 都 仔细 想 一 想 : 人 们 对 Java 做 了 大 量 的 工作 ， 使 它 能 方 
便 程 序 员 解决 应 用 级 问题 (如 连 网 和 跨 平 台 UI 等 ) ， 所 以 它 在 本 质 上 人 允 
许 人 们 创建 非常 大 型 和 有 灵活 的 代码 主体 。 同 时 ， 考 虑 到 Java 还 拥有 我 迄 
今 为 止 尚未 在 其 他 任何 一 种 语言 里 见 到 的 最 “健壮 ”的 类 型 检查 及 错误 控 
R A E E E 
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但 对 于 目 己 茶 个 特定 的 项 目 ， 真 的 可 以 不 假 思索 地 将 C++ 换 成 Java 吗 ? 
除了 Web 程 友 片 ， 还 有 两 个 问题 需要 考虑 。 痛 先 ， 假 如 要 使 用 大 量 现 有 
WE (这 样 肯 定 可 以 提高 不 少 的 效率 ) ， 或 者 已 经 有 了 一 个 坚实 的 C 或 
C++ 代码 库 ， 那 么 换 成 Java 后 ， 反 映 会 阻碍 开发 进度 ， 而 不 是 加 快 它 的 
速度 。 但 行 想 从 头 开 始 构建 目 己 的 所 有 人 代码， 那么 Java 的 简单 易 用 惑 能 
有 效 地 缩短 开发 时 间 。 





























最 大 的 问题 是 速度 。 在 原始 的 Java 解 释 占 中 ， 人 解释 过 的 Java 会 比 C 慢 上 

20 到 50 倍 。 尽 管 经 过 长 时 间 的 发 展 ， 这 个 速度 有 一 定 程度 的 提高 ， 但 和 
C 比 起 来 仍然 很 惹 殊 。 计 算 机 最 注重 的 就 是 速度 ， 假 如 在 一 台 计 算 机 上 
不 能 明显 较 快 地 干 活 ， 那 么 还 不 如 用 手 做 (有 人 建议 在 开发 期 间 使 用 

Java， 以 缩短 开发 时 间 。 然 后 用 一 个 工具 和 文 撑 库 将 代码 转换 成 C++， 

这 样 可 获得 更 快 的 执行 速度 ) 。 


为 使 Java 适 用 于 大 多 数 Web 开 发 项 目 ， 关 键 在 于 速度 上 的 改善 。 此 时 要 
用 到 人 们 称 为 “刚好 及 时 ”(Just-m Time， 或 JIT) 的 编译 器 ， 甚 至 考虑 
更 低级 的 代码 编译 器 《写作 本 书 时 ， 也 有 两 款 问 世 ) 。 当 然 ， 低 级 代码 
编译 器 会 使 编译 好 的 程序 不 能 路 平台 执行 ， 但 同时 也 带 来 了 速度 上 的 提 
升 。 这 个 速度 甚至 接近 C 和 C++。 而 且 Java 中 的 程序 交叉 编译 应 当 比 C 和 
C++ 中 简单 得 多 〈 理 论 上 只 需 重 编译 即 可 ， 但 实际 仍 较 难 实现 ， 其 他 语 
言 也 曾 作 出 类 似 的 保证 ) 。 


在 本 书 附录 ， 大 家 可 找到 与 Java / C++ 比较 .对 Java 现 状 的 观察 以 及 编 
码 规 则 有 关 的 内 容 。 
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第 2 章 一 切 都 是 对 象 
< 尽管 以 C++ 为 基础 ， 但 Java 是 一 种 更 纯粹 的 面向 对 象 程序 设计 语言 "。 


无 论 C++ 还 是 Java 都 属于 杂 合 语言 。 但 在 Java 中 ， 设 计 者 觉得 这 种 杂 合 
并 不 象 在 C++ 里 那么 重要 。 杂 合 语言 允许 采用 多 种 编程 风格 ; 之 所 以 说 
C++ 是 一 种 杂 合 语言 ， 是 因为 它 支 持 与 C 语 言 的 问 后 兼容 能 力 。 由 于 
C++ 是 C 的 一 个 超 集 ， 所 以 包含 的 许多 特性 都 是 后 者 不 具备 的 ， 这 些 特 
性 使 C++ 在 某 些 地 方 显得 过 于 复杂 。 


Java 语 言 首 先 便 假定 了 我 们 只 和 希望 进行 面 癌 对 象 的 程序 设计 。 也 就 是 
说 ， 正 式 用 它 设 计 之 前 ， 必 须 先 将 自己 的 思想 转 入 一 个 面 回 对 象 的 世界 

(除非 早已 习惯 了 这 个 世界 的 思维 方式 ) 。 只 有 做 好 这 个 准备 工作 ， 与 
其 他 OOP 语 言 相 比 ， 才 能 体会 到 Java 的 易学 易 用 。 在 本 章 ， 我 们 将 探讨 
的 基本 组 件 ， 并 体会 为 什么 说 Java 旋 至 Java 程 序 内 的 一 切 都 是 
XT R o 


























2.1 用 句柄 操纵 对 象 


每 种 编程 语言 都 有 目 己 的 数据 处 理 方 式 。 有 些 时 候 ， 程 序 员 必 须 时 刻 留 
意 准 备 处 理 的 是 什么 类 型 。 您 曾 利 用 一 些 特殊 语法 直接 操作 过 对 象 ， 或 
处 理 过 一 些 间 接 表 示 的 对 象 吗 〈C 或 C++ 里 的 指针 ) ? 


所 有 这 些 在 Java 里 都 得 到 了 简化 ， 任 何 东西 都 可 看 作对 象 。 因 此 ， 我 们 
可 采用 一 种 统一 的 语法 ， 任 何 地 方 均 可 照搬 不 误 。 但 要 注意 ， 尺 管 将 一 
切 都 “看 作 ” 对 象 ， 但 操纵 的 标识 从 实际 是 指 同 一 个 对 象 的 “ 句 

WA” (Handle) 。 在 其 他 Java 参 考 书 里 ， 还 可 看 到 有 的 人 将 其 称 作 一 

个 “引用 ”， 甚 至 一 个 “指针 ?”。 可 将 这 一 情形 想象 成 用 遥控 板 〈 句 柄 ) 操 
纵 电 视 机 (对象) 。 只 要 握 住 这 个 遥控 板 ， 就 相当 于 掌握 了 与 电视 机 连 
接 的 通道 。 但 一 旦 需要 “ 换 频 道 ” 或 者 “ 关 小 声 首 ”， 我 们 实际 操纵 的 是 路 
{zik CARA) ， 再 由 遥控 板 上 自己 操纵 电视 机 《〈 对 象 ) 。 如 果 要 在 房间 里 
ne 并 想 保 持 对 电视 机 的 控制 ， 那 么 手 上 拿 着 的 是 遥控 板 ， 而 非 
视 机 。 


此 外 ， 即 使 没有 电视 机 ， 遥 控 板 亦 可 独立 存在 。 也 就 是 说 ， 只 是 由 于 拥 
有 一 个 句柄 ， 并 不 表示 必须 有 一 个 对 象 同 它 连接 。 所 以 如 果 想 容纳 一 个 
词 或 句子 ， 可 创建 一 个 String 人 句柄 : 
































String S; 


但 这 里 创建 的 只 是 句柄 ， 并 不 是 对 象 。 夺 此 时 疝 s 发 送 一 条 消 恩 ， 束 会 
获得 一 个 错误 (运行 期 。 这 是 由 于 s 实 际 并 未 与 任何 东西 连接 〈 即 “ 没 
有 电视 机 ”) 。 因 此 ， 一 种 更 安全 的 做 法 是 : 创建 一 个 句柄 时 ， 记 住 无 
论 如 何 都 进行 初始 化 : 





String s = "asdf"; 


然而 ， 这 里 采用 的 是 一 种 特殊 类 型 : 字 串 可 用 加 引号 的 文字 初始 化 。 通 
常 ， 必 须 为 对 象 使 用 一 种 更 通用 的 初始 化 类 型 。 


2.2 所 有 对 象 部 必须 创建 


创建 句柄 时 ， 我 们 希望 它 同 一 个 新 对 象 连接 。 通 常用 new 关 键 字 达 到 这 
一 目的 。new 的 意思 是 :“ 把 我 变 成 这 些 对 象 的 一 种 新 类 型 >。 所 以 在 上 
面 的 例子 中 ， 可 以 说 : 


String s = new String("asdf"); 


它 不 仅 指 出 “将 我 变 成 一 个 新 字 串 ”"， 也 通过 提供 一 个 初始 字 串 ， 指 出 
了 “如 何 生成 这 个 新 字 串 ”。 


“A, FE CString) 并 非 唯一 的 类 型 。Java 配 套 提供 了 数量 众多 的 现 
成 类 型 。 对 我 们 来 讲 ， 最 重要 的 就 是 记 住 能 自行 创建 类 型 。 事 实 上 ， 这 
应 是 Java 程 序 设 计 的 一 项 基本 操作 ， 是 继续 本 书后 余部 分 学 习 的 基础 。 


2.2.1 保存 到 什么 地 方 


程序 运行 时 ， 我 们 最 好 对 数据 保存 到 什么 地 方 做 到 心中 有 数 。 特 别 要 注 
意 的 是 内 存 的 分 配 。 有 六 个 地 方 都 可 以 保存 数据 : 


(1) 寄存 器 。 这 是 最 快 的 保存 区 域 ， 因 为 它 位 于 和 其 他 所 有 保存 方式 不 
同 的 地 方 : 处 理 需 内 部 。 然 而 ， 寄 存 器 的 数量 十 分 有 限 ， 所 以 寄存 需 是 
根据 需要 由 编译 器 分 配 。 我 们 对 此 没有 直接 的 控制 权 ， 也 不 可 能 在 目 己 
的 程序 里 找到 寄存 器 存在 的 任何 踩 迹 。 


(2) 堆栈 。 驻 留 于 常规 RAM《 随 机 访问 存储 器 ) 区 域 ， 但 可 通过 它 的 “ 推 
栈 指 针 ? 获 得 处 理 的 直接 文 持 。 扒 栈 指针 吞 癌 下 移 ， 会 创建 新 的 内 存 ; 

奉 问 上 移 ， 则 会 释放 那些 内 存 。 这 是 一 种 特别 快 、 特 别 有 效 的 数据 保存 
方式 ， 仅 次 于 寄存 器。 创建 程序 时 ，Java 编 译 器 必须 准确 地 知道 堆栈 内 
保存 的 所 有 数据 的 “长 度 ” 以 及 “存在 时 间 ”。 这 是 由 于 它 必 须 生 成 相应 的 
代码 ， 以 便 向 上 和 向 下 移动 指针 。 这 一 限制 无 疑 影 响 了 程序 的 灵活 性 ， 
所 以 尽管 有 些 Java 数 据 要 保存 在 堆栈 里 一 一 特别 是 对 象 句柄 ， 但 Java 对 
象 并 不 放 到 其 中 。 


(3) 堆 。 一 种 常规 用 途 的 内 存 池 〔 也 在 RAM 区 域 》， 其 中 保存 了 Java 对 
Be MERRE, “WIRA” (Heap) 最 吸引 人 的 地 方 在 于 编译 器 
不 必 知道 要 从 堆 里 分 配 多 少 存储 空间 ， 也 不 必 知 道 存储 的 数据 要 在 堆 里 

















停留 多 长 的 时 间 。 因 此 ， 用 堆 保 存 数据 时 会 得 到 更 大 的 灵活 性 。 要 求 创 
律 一 个 对 象 时 ， 只 需 用 new 命 令 编制 相关 的 代码 即 可 。 执 行 这 些 代码 
时 ， 会 在 堆 里 自动 进行 数据 的 保存 。 当 然 ， 为 达到 这 种 灵活 性 ， 必 然 会 
付出 一 定 的 代价 :在 堆 里 分 配 存储 空间 时 会 花 掉 更 长 的 时 间 ! 


(4) 静态 存储 。 这 儿 的 “静态 ”(〈Static) 是 指 “ 位 于 固定 位 置 ”( 尽 管 也 在 
RAM 里 ) 。 程 序 运行 期 间 ， 静 态 存储 的 数据 将 随时 等 候 调 用 。 可 用 
static 关 键 字 指出 一 个 对 象 的 特定 元 素 是 静态 的 。 但 Java 对 象 本 号 永 远 都 
不 会 置 入 静态 存储 空间 。 


(5) 常数 存储 。 常 数值 通常 直接 置 于 程序 代码 内 部 。 这 样 做 是 安全 的 ， 
因为 它们 永远 都 不 会 改变 。 有 的 常数 需要 严格 地 保护 ， 所 以 可 考虑 将 它 
们 置 入 只 读 存储 器 (ROM) 。 


(6) 非 RAM 存 储 。 若 数据 完全 独立 于 一 个 程序 之 外 ， 则 程序 不 运行 时 仍 
可 存在 ， 并 在 程序 的 控制 范围 之 外 。 其 中 两 个 最 主要 的 例子 便 是 “ 流 式 
对 象 " 和 “固定 对 象 ”。 对 于 流 式 对 象 ， 对 象 会 变 成 字 节 流 ， 通 常会 发 给 

另 一 台 机 器 。 而 对 于 固定 对 象 ， 对 象 保存 在 磁盘 中 。 即 使 程序 中 止 运 

行 ， 它 们 仍 可 保持 自己 的 状态 不 变 。 对 于 这 些 类 型 的 数据 存储 ， 一 个 特 
别 有 用 的 技巧 就 是 它们 能 存在 于 其 他 媒体 中 。 一 旦 需要 ， 其 至 能 将 它们 
恢复 成 普通 的 、 基 于 RAM 的 对 象 。Java 1.1 提 供 了 对 Lightweight 
persistence 的 文 持 。 未 来 的 版 本 甚至 可 能 提供 更 完整 的 方案 。 


2.2.2 特殊 情况 : 主要 类 型 


有 一 系列 类 需 特 别 对 待 ， 可 将 它们 想象 成 “基本 、“ 主 要 ”或 

者 “ 主 ”(Primitive〉 类 型 ， 进 行程 序 设 计时 要 频繁 用 到 它们 。 之 所 以 要 
特别 对 待 ， 是 由 于 用 new 创 建 对 象 “ 特 别 是 小 的 、 简 单 的 变量 ) 并 不 是 
非常 有 效 ， 因 为 new 将 对 象 置 于 " 推 ?里 。 对 于 这 些 类 型 ，Java 采 纳 了 与 C 
和 C++ 相同 的 方法 。 也 就 是 说 ， 不 是 用 new 创 建 变量 ， 而 是 创建 一 个 并 
非 句 柄 的 “上 自动 ”变量 。 这 个 变量 容纳 了 有 具体 的 值 ， 并 置 于 堆栈 中 ， 能 够 
更 高 效 地 存 取 。 


Java 决 定 了 每 种 主要 类 型 的 大 小 。 就 象 在 大 多 数 语言 里 那样 ， 这 些 大 小 


并 不 随 痢 机 器 结构 的 变化 而 变化 。 这 种 大 小 的 不 可 更 改正 是 Java 程 序 具 
有 很 强 移 植 能 力 的 原因 之 一 。 


| | 















































float [82-bit|IEEE754 |IEEE754 
double |64-bit|rEEE754 |IEEE754 


-| he | 


©: 到 Java 1.1 才 有 ，1.0 版 没有 。 
数值 类 型 全 部 是 有 符号 正 负 号) K, 所 以 不 必 费 劲 寻找 没有 符号 的 类 
型 。 


T 





主 数据 类 型 也 拥有 自己 的 “封装 器 ”(wrapper) 类 。 这 意味 着 假如 想 让 堆 
内 一 个 非 主 要 对 象 表示 那个 主 类 型 ， 就 要 使 用 对 应 的 封装 堪 。 例 如 : 


char c= 'x'; 
Character C = new Character('c’); 
也 可 以 直接 使 用 : 


Character C = new Character('x'); 


这 样 做 的 原因 将 在 以 后 的 章节 里 解释 。 
1. 高 精度 数字 


Java 1.1 增 加 了 两 个 类 ， 用 于 进行 高 精度 的 计算 : BigInteger 和 
BigDecimal。 尺 管 它们 大 致 可 以 划分 为 “封装 器 ”类 型 ， 但 两 者 都 没有 对 
应 的 “ 主 类 型 ”。 


这 两 个 类 都 有 自己 特殊 的 “方法 "， 对 应 于 我 们 针对 主 类 型 执行 的 操作 。 
也 就 是 说 ， 能 对 int 或 float 做 的 事情 ， 对 BigInteger 和 BigDecimal 一 样 可 以 
做 。 只 是 必须 使 用 方法 调用 ， 不 能 使 用 运算 符 。 此 外 ， 由 于 牵涉 更 多 ， 
所 以 运算 速度 会 慢 一 些 。 我 们 牺牲 了 速度 ， 但 换 来 了 精度 。 


BigInteger 文 持 任 意 精 度 的 整数 。 也 惑 是 次 ， 我 们 可 精确 表示 任意 大 小 
的 整数 值 ， 同 时 在 运算 过 程 中 不 会 丢失 任何 信息 。 


持 任意 精度 的 定点 数字 。 例 如 ， 可 用 它 进 行 精 确 的 币值 计 

















~ 用 这 两 个 类 时 可 选用 的 构建 器 和 方法 ， 请 自行 参考 联机 帮助 文 


2.2.3 Java 的 数组 


几乎 所 有 程序 设计 语言 都 支持 数组 。 在 C 和 C++ 里 使 用 数组 是 非常 危险 

的 ， 因 为 那些 数组 只 是 内 存 块 。 奉 程序 访问 自己 内 存 块 以 外 的 数组 ， 或 

i me 内 存 ( 属 于 常规 编程 错误 ) ， 会 产生 不 可 预测 的 后 
CERO) 。 


D: 在 C++ 里 ， 应 尽量 不 要 使 用 数组 ， 换 用 标准 模板 库 (Standard 
TemplateLibrary) 里 更 安全 的 容器 。 


Java 的 一 项 主要 设计 目标 就 是 安全 性 。 所 以 在 C 和 C++ 里 困扰 程序 员 的 

许多 问题 都 未 在 Java 里 重复 。 一 个 Java 可 以 保证 被 初始 化 ， 而 且 不 可 在 
它 的 范围 之 外 访问 。 由 于 系统 自动 进行 范围 检查 ， 所 以 必然 要 付出 一 些 
代价 针对 每 个 数组 ， 以 及 在 运行 期 间 对 索引 的 校 验 ， 都 会 造成 少量 的 
内 存 开 销 。 但 由 此 换 回 的 是 更 高 的 安全 性 ， 以 及 更 高 的 工作 效率 。 为 此 
付出 少许 代价 是 值得 的 。 








创建 对 象 数 组 时 ， 实 际 创建 的 是 一 个 句柄 数组 。 而 且 每 个 句柄 都 会 目 动 
初始 化 成 一 个 特殊 值 ， 并 带 有 自己 的 关键 字 : mull 〈 空 )。 一 旦 Java 看 
到 null， 就 知道 该 句柄 并 未 指 回 一 个 对 象 。 正 式 使 用 前 ， 必 须 为 每 个 名 
柄 都 分 配 一 个 对 象 。 耕 试图 使 用 依然 为 null 的 一 个 句柄 ， 就 会 在 运行 期 
报告 问题 。 因 此 ， 上 典型 的 数组 错误 在 Java 里 就 得 到 了 避免。 


也 可 以 创建 主 类 型 数组 。 同 样 地 ， 编 译 右 能 够 担保 对 它 的 初始 化 ， 因 为 
会 将 那个 数组 的 内 存 划分 成 零 。 


数组 问题 将 在 以 后 的 章节 里 详细 讨论 。 


2.3 绝对 不 要 清除 对 象 


在 大 多 数 程序 设计 语言 中 ， 变 量 的 “存在 时 间 ”(Lifetime》 一 直 是 程序 
员 需 要 着 重 考虑 的 问题 。 变 量 应 持续 多 长 的 时 间 ? 如 果 想 清除 它 ， 那 么 
何 时 进行 ? 在 变量 存在 时 间 上 纠缠 不 清 会 造成 大 量 的 程序 错误 。 在 下 面 
的 小 节 里 ， 将 阅 示 Java 如 何 帮助 我 们 完成 所 有 清除 工作 ， 从 而 极 大 了 简 
化 了 这 个 问题 。 


2.3.1 作用 域 


大 多 数 程序 设计 语言 都 提供 S EHER (Scope) 的 概念 。 对 于 在 作用 
域 里 定义 的 名 字 ， 作 用 域 同 时 决定 了 它 的 “可 见 性 ”以 及 “存在 时 间 ”。 在 
C，C++ 和 Java 里 ， 作 用 域 是 由 花 括 号 的 位 置 决定 的 。 参 考 下 面 这 个 例 




















{ 


int x = 12; 
/* only x available */ 
{ 
int q = 96; 
/* both x & q available */ 
} 
/* only x available */ 
/* q “out of scope” */ 


i 








作为 在 作用 域 里 定义 的 一 个 变量 ， 它 只 有 在 那个 作用 域 结 束 之 前 才 可 使 
用 。 


在 上 面 的 例子 中 ， 缩 进 排版 使 Java 代 码 更 易 辩 读 。 由 于 Java 是 一 种 形式 
目 由 的 语言 ， 所 以 额外 的 空格 、 制 表 位 以 及 回 车 都 不 会 对 结果 程序 造成 


影响 。 


注意 尽管 在 C 和 C++ 里 是 合法 的 ， 但 在 Java 里 不 能 象 下 面 这 样 书写 代 





int x = 12; 


int x = 96; /* illegal */ 





编译 器 会 认为 变量 x 已 被 定义 。 所 以 C 和 C++ 能 将 一 个 变量 “隐藏 "在 一 个 
更 大 的 作用 域 里 。 但 这 种 做 法 在 Java 里 是 不 允许 的 ， 因 为 Java 的 设计 者 
认为 这 样 做 使 程序 产生 了 混淆 。 

2.3.2 对 象 的 作用 域 

Java 对 象 不 具备 与 主 类 型 一 样 的 存在 时 则 。 用 new 关 键 字 创建 一 个 Java 
对 象 的 时 候 ， 它 会 超出 作用 域 的 范围 之 外 。 所 以 假 春 使 用 下 面 这 段 代 
hH: 

{ 

String s = new String("a string"); 

} /* 作用 域 的 终点 */ 

那么 句柄 s 会 在 作用 域 的 终点 处 消失 。 然 而 ，s 指 向 的 String 对 象 依然 占 


据 着 内 存 空间 。 在 上 和 面 这 段 代 码 里 ， 我 们 没有 办 法 访问 对 象 ， 因 为 指 问 
它 的 唯一 一 个 句柄 已 超出 了 作用 域 的 边界 。 在 后 面 的 章节 里 ， 大 家 还 会 











继续 学 习 如 何在 程序 运行 期 间 传递 和 复制 对 象 句柄 。 


这 样 造成 的 结果 便 是 : 对 于 用 new 创 建 的 对 象 ， 只 要 我 们 愿意 ， 它 们 就 
会 一 直 保留 下 去 。 这 个 编程 问题 在 C 和 C++ 里 特别 突出 。 看 来 在 C++ 里 
遇 到 的 麻烦 最 大 : 由 于 不 能 从 语言 获得 任何 帮助 ， 所 以 在 需要 对 象 的 时 
候 ， 根 本 无 法 确定 它们 是 否 可 用 。 而 且 更 麻烦 的 是 ， 在 C++ 里 ， 一 旦 工 
作 完 成 ， 必 须 保 证 将 对 象 清除 。 


这 样 便 带 来 了 一 个 有 趣 的 问题 。 假 如 Java 让 对 象 依然 故我 ， 怎 样 才 能 防 
止 它们 大 量 充 斥 内 存 ， 并 最 终 造 成 程序 的 “凝固 ?" 呢 。 在 C++ 里 ， 这 个 问 
题 最 令 程序 员 头 痛 。 但 Java 以 后 ， 情 况 却 发 生 了 改观 。Java 有 一 个 特别 
的 “垃圾 收集 右 ”*， 它 会 查找 用 new 创 建 的 所 有 对 象 ， 并 辨别 其 中 哪些 不 
再 被 引用 。 随 后 ， 它 会 自动 释放 由 那些 内 置 对 象 占据 的 内 存 ， 以 便 能 由 
新 对 象 使 用 。 这 意味 着 我 们 根本 不 必 操 心 内 存 的 回收 问题 。 只 需 简 单 地 
创建 对 象 ， 一 旦 不 再 需要 它们 ， 它 们 就 会 自动 离 去 。 这 样 做 可 防止 在 

een eine oe ee uae rene 

溢出 ”。 











2.4 新 建 数 据 交 型 : 类 


如 果 说 一 切 东 西 都 是 对 象 ， 那 么 用 什么 决定 一 个 “类 ”(Class) 的 外 观 与 
行为 呢 ?” 换 句 话说 ， 是 什么 建立 起 了 一 个 对 象 的 “类 型 "(Type) We? 大 
家 可 能 猜想 有 一 个 名 为 “type” 的 关键 字 。 但 从 历史 看 来 ， 大 多 数 面 问 对 
象 的 语言 都 用 关键 字 “class” 表 达 这 样 一 个 意思 : “我 准备 告诉 你 对 象 一 

种 新 类 型 的 外 观 ”。class 关 键 字 太 常用 了 ， 以 至 于 本 书 许多 地 方 并 没有 

人 在 这 个 关键 字 的 后 面 ， 应 该 跟随 新 数据 类 
3H Ko PU: 


class ATypeName {/* 类 主体 置 于 这 里 } 


eee 入 了 一 种 新 类 型 ， 接 下 来 便 可 用 new 创 建 这 种 类 型 的 一 个 新 对 


ATypeName a = new ATypeName(); 


在 ATypeName 里 ， 类 主体 只 由 一 条 注释 构成 ( 星 号 和 和 斜 杜 以 及 其 中 的 内 
容 ， 本 章 后 面 还 会 详细 讲述 ) ， 所 以 并 不 能 对 它 做 太 多 的 事情 。 事 实 
上 ， 除 非 为 其 定义 了 某 些 方法 ， 否 则 根本 不 能 指示 它 做 任何 事情 。 


2.4.1 字段 和 方法 


定义 一 个 类 时 〈 我 们 在 Java 里 的 全 部 工作 就 是 定义 类 、 制 作 那些 类 的 对 
象 以 及 将 消息 发 给 那些 对 象 ) ， 可 在 目 己 的 类 里 设置 两 种 类 型 的 元 素 : 

数据 成 员 《〈《 有 时 也 叫 “ 字 段 ?) 以 及 成 员 函 数 〈 通 常 叫 “方法 ”) 。 其 中 ， 

数据 成 员 是 一 种 对 象 〈 通 过 它 的 句柄 与 其 通信 ) ， 可 以 为 任何 类 型 。 它 
也 可 以 是 主 类 型 〈 并 不 是 句柄 ) 之 一 。 如 果 是 指向 对 象 的 一 个 句柄 ， 则 
必须 初始 化 那个 句柄 ， 用 一 种 名 为 “构建 器 ”《〈 第 4 章 会 对 此 详 述 ) 的 特 
殊 函 数 将 其 与 一 个 实际 对 象 连接 起 来 “就 象 早先 看 到 的 那样 ， 使 用 new 
REF) 。 但 大 是 一 种 主 类 型 ， 则 可 在 类 定义 位 置 直接 初始 化 (正如 后 
面 会 看 到 的 那样 ， 句 柄 亦 可 在 定义 位 置 初 始 化 ) 。 


每 个 对 象 都 为 目 己 的 数据 成 员 保 有 存储 空间 ;数据 成 员 不 会 在 对 象 之 间 
共 诗 。 下 面 是 定义 了 一 些 数 据 成 员 的 类 示例 : 


class DataOnly { 














int 1; 
float f; 


boolean b; 


这 个 类 并 没有 做 任何 实质 性 的 事情 ， 但 我 们 可 创建 一 个 对 象 : 

DataOnly d = new DataOnly(); 

可 将 值 赋 给 数据 成 员 ， 但 首先 必须 知道 如 何 引 用 一 个 对 象 的 成 员 。 为 达 

到 引用 对 象 成 员 的 目的 ， 首 先 要 写 上 对 象 句柄 的 名 字 ， 再 跟随 一 个 点 号 
(句点 ) ， 再 跟随 对 象 内 部 成 员 的 名 字 。 即 “对 象 句 柄 .成 员 ”。 例 如 

d.i = 47; 

d.f = 1.1f; 

d.b = false; 


一 个 对 象 也 可 能 包含 了 为 一 个 对 象 ， 而 妨 一 个 对 象 里 则 包含 了 我 们 想 修 
改 的 数据 。 对 于 这 个 问题 ， 只 需 保持 “连接 句点 ” 即 可 。 例 如 : 


myPlane.leftTank.capacity = 100; 

除 容纳 数据 之 外 ，DataOnly 类 再 也 不 能 做 更 多 的 事情 ， 因 为 它 没 有 成 员 
函数 (方法 ) 。 为 正确 理解 工作 原理 ， 首 先 必须 知道 “ 自 变 量 " 和 “返回 
值 ”的 概念 。 我 们 号 上 束 会 详 加 解释 。 

1. 主 成 员 的 默认 值 


右 某 个 主 数据 类 型 属于 一 个 类 成 员 ， 那 么 即使 不 明确 《〈“ 显 式 ) 进行 初始 
化 ， 也 可 以 保证 它们 获得 一 个 默认 值 。 


主 类 型 默认 值 





Boolean false 

Char \u0000'(null) 

byte (byte)0 

short (Short)0 

int 0 

long OL 

float 0.0f 

double 0.0d 

一 旦 将 变量 作为 类 成 员 使 用 ， 就 要 特别 注意 由 Java 分 配 的 默认 值 。 这 样 
做 可 保证 主 类 型 的 成 员 变 量 肯 定 得 到 了 初始 化 〈C++ 不 具备 这 一 功 
fe) ， 可 有 效 过 止 多 种 相关 的 编程 错误 。 


然而 ， 这 种 保证 却 并 不 适用 于 “局 部 ”变量 一 一 那些 变量 并 非 一 个 类 的 字 
段 。 所 以 ， 假 大 在 一 个 函数 定义 中 写 入 下 述 代码 : 














int x; 


那么 x 会 得 到 一 些 随机 值 〈 这 与 C 和 C++ 是 一 样 的 ) ， 不 会 自动 初始 化 成 
零 。 我 们 责任 是 在 正式 使 用 x 前 分 配 一 个 适当 的 值 。 如 果 态 记 ， 就 会 得 
到 一 条 编译 期 错误 ， 告 诉 我 们 变量 可 能 尚未 初始 化 。 这 种 处 理 正 是 Java 
优 于 C++ 的 表现 之 一 。 许 多 C++ 编译 器 会 对 变量 未 初始 化 发 出 警告 ， 但 
在 Java 里 却 是 错误 。 


2.5 方法 、 目 变量 和 返回 值 


SAIL, RM—-BA“e ee’ (Function) 这 个 词 指 代 一 个 已 命 pany fF 
PFE o rae 更 第 用 的 一 个 词 却 是 HIE” (Method) ， 代 表 “ 完 
成 某 事 的 途径 >。 尽 管 它们 表达 的 实际 是 同一 个 意思 ， 但 从 现在 开始 ， 
本 书 将 一 直 使 用 方法" 而 不 是 “函数 ”。 


Java 的 “方法 ?决定 了 一 个 对 象 能 够 接收 的 消息 。 通 过 本 节 的 学 习 ， 大 家 
会 知道 方法 的 定义 有 多 么 简单 ! 


方法 的 基本 组 成 部 分 包括 名 字 、 目 变量 、 返 回 类 型 以 及 主体 。 下 面 便 是 
它 最 基本 的 形式 .: 


返回 类 型 方法 名 ( /* 自 变量 列表 */ ) {/* 方法 主体 */} 


返回 类 型 是 指 调用 方法 之 后 返回 的 数值 类 型 。 显 然 ， 方 法 名 的 作用 是 对 
ee 目 变 量 列表 列 出 了 想 传递 给 方法 的 信息 类 
型 和 名 称 


Java 的 方法 只 能 作为 类 的 一 部 分 创建 。 只 能 针对 某 个 对 象 调 用 一 个 方法 
(注释 @)) ， 而 且 那 个 对 象 必 须 能 够 执行 那个 方法 调用 。 和 若 试图 为 一 个 
对 象 调用 错误 的 方法 ， 就 会 在 编译 期 得 到 一 条 出 错 消息 。 为 一 个 对 象 调 
用 方法 时 ， 需 要 先 列 出 对 象 的 名 字 ， 在 后 面 跟 上 一 Sale, FRERL AVE 
名 以 及 它 的 参数 列表 。 亦 即 “ 对 象 名 .方法 名 ( 自 变 量 1， 自 变量 2， 自 变量 
3...)。 举 个 例子 来 说 ， 假 设 我 们 有 一 个 方法 名 叫 f()， 它 没有 自 变 量 ， 返 
回 的 是 类 型 为 int 的 一 个 值 。 那 么 ， 假 设 有 一 个 名 为 8 的 对 象 ， 可 为 其 调 
用 方法 {0， 则 代码 如 下 : 


int x = a.fQ); 

返回 值 的 类 型 必须 兼容 x 的 类 型 。 

象 这 样 调用 一 个 方法 的 行动 通常 叫 作 “ 向 对 象 发 送 一 条 消 上 号 ”。 在 上 面 的 
例子 中 ， 消 息 是 DO， 而 对 象 是 a。 面 问 对 象 的 程序 设计 通 重 常 简单 地 归纳 
ASAT RAGE SY” 


©: 正如 马上 就 要 学 到 的 那样 , “HASTE Ra, H i PT 












































象 。 
2.5.1 自 变 量 列表 


目 变 量 列表 规定 了 我 们 传送 给 方法 的 是 什么 信息 。 正 如 大 家 或 许 已 猜 到 
的 那样 ， 这 些 信息 一 一 如 同 Java 内 其 他 任何 东西 一 一 采用 的 都 是 对 象 的 
形式 。 因 此 ， 我 们 必须 在 自 变 量 列表 里 指定 要 传递 的 对 象 类 型 ， 以 及 每 
个 对 象 的 名 字 。 正 如 在 Java 其 他 地 方 处 理 对 象 时 一 样 ， 我 们 实际 传递 的 
是 “句柄 ”( 注 释 由 ) 。 然 而 ， 句 柄 的 类 型 必须 正确 。 倘 知 希 望 自 变量 是 
一 个 “ 字 串 ”， 那 么 传递 的 必须 是 一 个 字 串 。 


O: 对 于 前 面 提 及 的 “特殊 ”数据 类 型 boolean，char，byte，short，int， 
long，，float 以 及 double 来 说 是 一 个 例外 。 但 在 传递 对 象 时 ， 通 常 都 是 
指 传递 指向 对 象 的 句柄 。 


下 面 让 我 们 考虑 将 一 个 字 串 作为 目 变 量 使 用 的 方法 。 下 面 列 出 的 是 定义 
代码 ， 必 须 将 它 置 于 一 个 类 定义 里 ， 否 则 无 法 编译 : 





























int storage(String s) { 
return s.length() * 2; 
} 


这 个 方法 告诉 我 们 需要 多 少 字 节 才 外 容纳 一 个 特定 字 串 里 的 信息 (CFE 
里 的 每 个 字符 都 是 16 位 ， 或 者 说 2 个 字 节 、 长 整数 ， 以 便 提供 对 Unicode 
字符 的 支持 ) 。 自 变量 的 类 型 为 String， 而 且 叫 作 s。 一 旦 将 传递 给 方 

法 ， 就 可 将 它 当 作 其 他 对 象 一 样 处 理 (可 向 其 发 送 消息 ) 。 在 这 里 ， 我 
们 调用 的 是 length0) 方 法 ， 它 是 String 的 方法 之 一 。 该 方法 返回 的 是 一 个 
字 串 里 的 字符 数 。 


通过 上 面 的 例子 ， 也 可 以 了 解 retum 关 键 字 的 运用 。 它 主要 做 两 件 事 
情 。 首 先 ， 它 意味 着 “离开 方法 ， 我 已 完工 了 ”。 其 次 ， 假 设 方法 生成 了 
一 个 值 ， 则 那个 值 紧 接 在 retum 语 句 的 后 面 。 在 这 种 情况 下 ， 返 回 值 是 
通过 计算 表达 式 “s.length(*2” 而 产生 的 。 


Se E 回 任 意 类 型 ， 但 倘若 不 想 返 回 任何 东西 ， 束 可 指示 方 
法 返回 void F) 。 下 面 列 出 一 些 例 子 。 











boolean flag() { return true; } 

float naturalLogBase() { return 2.718; } 
void nothing() { return; } 

void nothing2() {} 


FREIA Avoid, Wlretun j HE — AY VE FS B.A HE. 
抵达 方法 末尾 ， 该 关键 字 便 不 需要 了 。 可 在 任何 地 方 从 一 个 方法 返回 。 
但 假设 已 指定 了 一 种 非 void 的 返回 类 型 ， 那 么 无 论 从 何 地 返回 ， 编 译 右 
都 会 确保 我 们 返回 的 是 正确 的 类 型 。 


PEKE, 大 家 或 许 已 得 到 了 这 样 的 一 个 印象 :一 个 程序 只 是 一 系列 对 
象 的 集合 ， 它 们 的 方法 将 其 他 对 象 作为 自己 的 日 变 量 使 用 ， 而 且 将 消息 
发 给 那些 对 象 。 这 种 说 法 大 体 正确 ， 但 通过 以 后 的 学 习 ， 大 家 还 会 知道 
如 何在 一 个 方法 里 作出 决策 ， 做 一 些 更 细致 的 基层 工作 。 至 于 这 一 章 ， 
只 需 理 解 消息 传送 就 足够 了 。 














2.6 构建 Java 程 序 
正式 构建 自己 的 第 一 个 Java 程 序 前 ， 还 有 几 个 问题 需要 注意 。 
2.6.1 名 字 的 可 见 性 


在 所 有 程序 设计 语言 里 ， 一 个 不 可 避免 的 问题 是 对 名 字 或 名 称 的 控制 。 
假设 您 在 程序 的 茶 个 模块 里 使 用 了 一 个 名 字 ， 而 另 一 名 程序 员 在 另 一 个 
模块 里 使 用 了 相同 的 名 字 。 此 时 ， 如 何 区 分 两 个 名 字 ， 并 防止 两 个 名 字 
互相 冲突 呢 ? 这 个 问题 在 C 语 言 里 特别 突出 。 因 为 程序 未 提供 很 好 的 名 
字 管 理 方 法 。C++ 的 类 【〔 即 Java 类 的 基础 〉 髓 套 使 用 类 里 的 函数 ， 使 其 
不 至 于 同 其 他 类 里 的 验 套 函数 名 冲突 。 然 而 ，C++ 仍 然 允 许 使 用 全 局 数 
据 以 及 全 局 函数 ， 所 以 仍然 难以 避免 冲突 。 为 解决 这 个 问题 ，C++ 用 额 
外 的 关键 字 引 入 了 “命名 空间 ”的 概念 。 


由 于 采用 全 新 的 机 制 ， 所 以 Java 能 完全 避免 这 些 问题 。 为 了 给 一 个 库 生 
成 明确 的 名 字 ， 采 用 了 与 Internet 域 名 类 似 的 名 字 。 事 实 上 ，Java 的 设计 
者 鼓励 程序 员 反 转 使 用 自己 的 Intemet 域 名 ， 因 为 它们 肯定 是 独一无二 
的 。 由 于 我 的 域名 是 BruceEckel.com， 所 以 我 的 实用 工具 库 就 可 命名 为 
com.bruceeckel.utility.foibles。 反 转 了 域名 后 ， 可 将 点 号 想象 成 子 目 录 。 


在 Java 1.0 和 Java 1.1 中 ， 域 扩展 名 com，edu，org，net 等 都 约定 为 大 写 
形式 。 所 以 库 的 样子 就 变 成 :COM.bruceeckel.utility.foibles。 然 而 ， 在 
Java 1.2 的 开发 过 程 中 ， 设 计 者 发 现 这 样 做 会 造成 一 些 问 题 。 所 以 目前 
的 整个 软件 包 都 以 小 写字 母 为 标准 。 


Java 的 这 种 特殊 机 制 意味 着 所 有 文件 都 自动 存在 于 自己 的 命名 空间 里 。 
而 且 一 个 文件 里 的 每 个 类 都 日 动 获得 一 个 独一无二 的 标识 符 〈 当 然 ， 一 
个 文件 里 的 类 名 必须 是 唯一 的 ) 。 所 以 不 必 学 习 特 殊 的 语言 知识 来 解决 
这 个 问题 一 一 语言 本 身 己 帮 我 们 照顾 到 这 一 点 。 


2.6.2 使 用 其 他 组 件 


一 旦 要 在 目 己 的 程序 里 使 用 一 个 预先 定义 好 的 类 ， 编 诺 器 就 必须 知道 如 
何 找到 它 。 当 然 ， 这 个 类 可 能 惑 在 发 出 调用 的 那个 相同 的 源码 文件 里 。 
如 果 是 那 种 情况 ， 只 需 简 单 地 使 用 这 个 类 即 可 一 一 即使 它 直 到 文件 的 后 
面 仍 未 得 到 定义 。Java 消 除了 “向 前 引用 ”的 问题 ， 所 以 不 要 关心 这 些 事 












































情 。 


但 假若 那个 类 位 于 其 他 文件 里 昵 ? 您 或 许 认 为 编译 器 应 该 足够 “联盟 ”， 
可 以 自行 发 现 它 。 但 实情 并 非 如 此 。 假 设 我 们 想 使 用 一 个 具有 特定 名 称 
的 类 ， 但 那个 类 的 定义 位 于 多 个 文件 里 。 或 者 更 粳 ， 假 设 我 们 准备 写 一 
个 程序 ， 但 在 创建 它 的 时 候 ， 却 同上 自己 的 库 加 入 了 一 个 新 类 ， 它 与 现 有 
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为 解决 这 个 问题 ， 必 须 消 除 所 有 湾 在 的、 纠缠 不 清 的 情况 。 为 达到 这 个 
目的 ， 要 用 import 关 键 字 准确 告诉 Java 编 译 器 我 们 希望 的 类 是 什么 。 
import 的 作用 是 指示 编译 器 导入 一 个 “ 包 ” 或 者 说 一 个 “类 库 ” (在 其 
他 语言 里 ， 可 将 “ 库 ? 想 象 成 一 系列 函数 、 数 据 以 及 类 的 集合 。 但 请 记 
住 ，Java 的 所 有 代码 都 必须 写 入 一 个 类 中 ) 。 


大 多 数 时 候 ， 我 们 直接 采用 来 自 标准 Java 库 的 组 件 〈 部 件 ) 即 可 ， 它 们 
是 与 编译 器 配套 提供 的 。 使 用 这 些 组 件 时 ， 没 有 必要 关心 元 长 的 保留 域 
名 ; 举 个 例子 来 说 ， 只 需 象 下 面 这 样 写 一 行 代码 即 可 : 














import java.util.Vector; 


它 的 作用 是 告诉 编译 器 我 们 想 使 用 Java 的 Vector 类 。 然 而 ，util 包 含 了 数 
量 众 多 的 类 ， 我 们 有 时 希望 使 用 其 中 的 几 个 ， 同 时 不 想 全 部 明确 地 声明 
它们 。 为 达到 这 个 目的 ， 可 使 用 “*” 通 配 符 。 如 下 所 示 : 


import java.util.*; 


再 导入 一 系列 类 时 ， 采 用 的 通常 是 这 个 办 法 。 应 尽量 避免 一 个 一 个 地 导 
入 类 。 


2.6.3 static 关 键 字 


通常 ， 我 们 创建 类 时 会 指出 那个 类 的 对 象 的 外 观 与 行为 。 除 非 用 new 创 
建 那个 类 的 一 个 对 象 ， 否 则 实际 上 并 未 得 到 任何 东西 。 只 有 执行 了 new 
后 ， 才 会 正式 生成 数据 存储 空间 ， 并 可 使 用 相应 的 方法 。 


但 在 两 种 特殊 的 情形 下 ， 上 述 方法 并 不 堪 用 。 一 种 情形 是 只 想 用 一 个 存 
储 区 域 来 保存 一 个 特定 的 数据 一 一 无 论 要 创建 多 少 个 对 象 ， 甚 至 根本 不 
创建 对 象 。 力 一 种 情形 是 我 们 主要 一 个 特殊 的 方法 ， 它 没有 与 这 个 类 的 





任何 对 象 天 联 。 也 就 是 说 ， 即 使 没有 创建 对 象 ， 也 需要 一 个 能 调用 的 方 
法 。 为 满足 这 两 方面 的 要 求 ， 可 使 用 static (AAS) 关键 字 。 一 旦 将 什么 
东西 设 为 static， 数 据 或 方法 就 不 会 同 那 个 类 的 任何 对 象 实例 联系 到 一 
起 。 所 以 尽管 从 未 创建 那个 类 的 一 个 对 象 ， 仍 能 调用 一 个 static 方 法 ， 或 
访问 一 些 static 数 据 。 而 在 这 之 前 ， 对 于 非 static 数 据 和 方法 ， 我 们 必须 
创建 一 个 对 象 ， 并 用 那个 对 象 访问 数据 或 方法 。 这 是 由 于 非 static 数 据 和 
方法 必须 知道 它们 操作 的 具体 对 象 。 当 然 ， 在 正式 使 用 前 ， 由 于 static 方 
法 不 需要 创建 任何 对 象 ， 所 以 它们 不 可 简单 地 调用 其 他 那些 成 员 ， 同 时 
不 引用 一 个 已 命名 的 对 象 ， 从 而 直接 访问 非 static 成 员 或 方法 (因为 非 
static 成 员 和 方法 必须 同一 个 特定 的 对 象 关 联 到 一 起 ) 。 

有 些 面 同 对 象 的 语言 使 用 了 “类 数据 ”和 “类 方法 ”这 两 个 术语 。 它 们 意味 
着 数据 和 方法 只 是 为 作为 一 个 整体 的 类 而 存在 的 ， 并 不 是 为 那个 类 的 任 
何 特定 对 象 。 有 时 ， 您 会 在 其 他 一 些 Java 书 刊 里 发 现 这 样 的 称呼 


为 了 将 数据 成 员 或 方法 设 为 static， 只 需 在 定义 前 置 和 这 个 关键 字 即 可 。 
例如 ， 下 述 代码 能 生成 一 人 static 数 据 成 员 ， 并 对 其 初始 化 : 


class StaticTest { 











Static int i = 47; 
} 


现在 ， 尽 管 我 们 制作 了 两 个 StaticTest 对 象 ， 但 它们 仍然 只 占据 
StaticTest.i 的 一 个 存储 空间 。 这 两 个 对 象 都 共享 同样 的 i。 请 考察 下 述 代 


StaticTest stl = new StaticTest(); 
StaticTest st2 = new StaticTest(); 


此 时 ， 无 论 st1.i 还 是 st2.i 都 有 同样 的 值 47， 因 为 它们 引用 的 是 同样 的 内 
存 区 域 。 


有 了 两 个 办 法 可 引用 一 个 static 变 量 。 正 如 上 面 展示 的 那样 ， 可 通过 一 个 对 
象 命名 它 ， 如 st2.i。 亦 可 直接 用 它 的 类 名 引用 ， 而 这 在 非 静 态 成 员 里 是 
行 不 通 的 (最 好 用 这 个 办 法 引用 static 变 量 ， 因 为 它 强 调 了 那个 变量 











的 “静态 ”本 质 ) 
StaticTest.i++; 
其 中 ，++ 运 算 符 会 使 变量 增值 。 此 时 ， 无 论 stLi 还 是 st2.i 的 值 都 是 48。 


类 似 的 馆 辑 也 适用 于 静态 方法 。 既 可 象 对 其 他 任何 方法 那样 通过 一 个 对 
象 引用 静态 方法 ， 亦 可 用 特殊 的 语法 格式 “类 名 .方法 0" 加 以 引用 。 毅 态 
方法 的 定义 是 类 似 的 : 


class StaticFun { 
static void incr() { StaticTest.i++; } 
} 


从 中 可 看 出 ，StaticFun 的 方法 incr(0) 使 静态 数据 i 增值 。 通 过 对 象 ， 可 用 
典型 的 方法 调用 incrO): 


StaticFun sf = new StaticFun(); 

sf.incr(); 

或 者 ， 由 于 incrO0 是 一 种 静态 方法 ， 所 以 可 通过 它 的 类 直接 调用 : 
StaticFun.incr(); 


尽管 是 “静态 ”的 ， 但 只 要 应 用 于 一 个 数据 成 员 ， 束 会 明确 改变 数据 的 创 
建 方式 (一 个 类 一 个 成 员 ， 以 及 每 个 对 象 一 个 非 静 态 成 员 ) 。 厦 应 用 于 
一 个 方法 ， 束 没有 那么 戏剧 化 了 。 对 方法 来 说 ，static 一 项 重要 的 用 途 束 
古 帮 助 我 们 在 不 必 创 建 对 象 的 前 提 下 调用 那个 方法 。 正 如 以 后 会 看 到 的 
a -点 是 至 关 重 要 的 一 一 特别 是 在 定义 程序 运行 入 口 方 法 main() 
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和 其 他 任何 方法 一 样 ，static 方 法 也 能 创建 自己 类 型 的 命名 对 象 。 所 以 经 
， 把 static 方 法 作为 一 个 “领头 羊 ? 使 用 ， 用 它 生成 一 系列 目 己 类 型 的 “ 实 
列 ”。 











2.7 我 们 的 第 一 个 Java 程 序 


最 后 ， 让 我 们 正式 编 一 个 程序 注释) 。 它 能 打印 出 与 当前 运行 的 系 
统 有 关 的 资料 ， 并 利用 了 来 自 Java 标 准 库 的 System 对 象 的 多 种 方法 。 注 
意 这 里 引入 了 一 种 额外 的 注释 样式 :“//*。 它 表示 到 本 行 结束 前 的 所 有 
内 容 部 是 注释 : 


// Property.java 





import java.util.*; 
public class Property { 
public static void main(String[] args) { 
System.out.println(new Date()); 
Properties p = System.getProperties(); 
p.list(System.out); 
System.out.println("--- Memory Usage:"); 
Runtime rt = Runtime.getRuntime(); 
System.out.println("Total Memory = " 
+ rt.totalMemory() 
+ " Free Memory = " 


+ rt.freeMemory()); 





©: 在 茶 些 编程 环境 里 ， 程 序 会 在 屏幕 上 一 切 而 过 ， 甚 至 没 机 会 看 到 结 
果 。 可 将 下 面 这 段 代 码 置 于 main() 的 末尾 ， 用 它 暂 停 输 出 : 


try { 

Thread.currentThread().sleep(5 * 1000); 
} catch(InterruptedException e) {} 

} 


它 的 作用 是 暂停 输出 5 秒 钟 。 这 段 代码 涉及 的 一 些 概念 要 到 本 书后 面 才 
会 讲 到 。 所 以 目前 不 必 深 究 ， 只 知道 它 是 让 程序 暂停 的 一 个 技巧 便 可 。 


在 每 个 程序 文件 的 开头 ， 都 必须 放置 一 个 import 语 句 ， 导 入 那个 文件 的 
代码 里 要 用 到 的 所 有 额外 的 类 。 注 意 我 们 说 它们 是 “额外 ”的 ， 因 为 一 个 
特殊 的 类 库 会 自动 导入 每 个 Java 文 件 : java.lang。 启 动 您 的 Web 浏 览 

器 ， 查 看 由 Sun 提 供 的 用 户 文 档 (如 果 尚 未 从 http://www.java.sun.com 下 
载 ， 或 用 其 他 方式 安装 了 Java 文 要 ， 请 立即 下 载 ) 。 在 packages.html 文 
件 里 ， 可 找到 Java 配 套 提供 的 所 有 类 库 名 称 。 请 选择 其 中 的 java.lang。 
在 “Class Index” 下 面 ， 可 找到 属于 那个 库 的 全 部 类 的 列表 。 由 于 
java.lang 默 认 进 入 每 个 Java 代 码 文 件 ， 所 以 这 些 类 在 任何 时 候 都 可 直接 
使 用 。 在 这 个 列表 里 ， 可 发 现 System 和 Runtime， 我 们 在 Property.java 里 
用 到 了 它们 。java.lang 里 没有 列 出 Date 类 ， 所 以 必须 导入 另 一 个 类 库 才 
能 使 用 它 。 如 果 不 清楚 一 个 特定 的 类 在 哪个 类 库 里 ， 或 者 想 检 视 所 有 的 
类 ， 可 在 Java 用 户 文 档 里 选择 “Class Hierarchy”( 类 分 级 结构 ) 。 在 Web 
浏览 器 中 ， 虽 然 要 花 不 短 的 时 间 来 建立 这 个 结构 ， 但 可 清楚 找到 与 Java 
配套 提供 的 每 一 个 类 。 随 后 ， 可 用 浏览 器 的 “人 查找”(Find) 功能 搜索 关 
键 字 “Date”。 经 这 样 处 理 后 ， 可 发 现 我 们 的 搜索 目标 以 java.util.Date 的 形 
式 列 出 。 我 们 终于 知道 它 位 于 util 库 里 ， 所 以 必须 导入 javautil*; 否则 
便 不 能 使 用 Date。 


观察 packages.html 文 档 最 开头 的 部 分 〈 我 已 将 其 设 为 自己 的 默认 起 始 
Tl) ， 请 选择 java.lang， 再 选 System。 这 时 可 看 到 System 类 有 几 个 字 
段 。 奉 选择 out， 就 可 知道 它 是 一 个 static PrintStream 对 象 。 由 于 它 是 二 
态 ” 的 ， 所 以 不 需要 我 们 创建 任何 东西 。out 对 象 表 定 是 3， 所 以 只 需 直 
接 用 它 即 可 。 我 们 能 对 这 个 out 对 象 做 的 事情 由 它 的 类 型 决定 : 
PrintStream。PrintStream 在 说 明文 字 中 以 一 个 超 链接 的 形式 列 出 ， 这 一 
点 做 得 非常 方便 。 所 以 假 知 单 击 那个 链接 ， 就 可 看 到 能 够 为 PrintStream 
调用 的 所 有 方法 。 方 法 的 数量 不 少 ， 本 书后 面 会 详细 介绍 。 就 目前 来 
说 ， 我 们 感 兴趣 的 只 有 printtnO0。 它 的 意思 是 “把 我 给 你 的 东西 打印 到 控 














制 台 ， 并 用 一 个 新 行 结束 ”。 所 以 在 任何 Java 程 序 中 ， 一 旦 要 把 某 些 内 
容 打印 到 控制 侣 ， 束 可 条 件 反 射 地 写 上 System.out.println(" 内 容 ")。 


类 名 与 文件 是 一 样 的 。 知 象 现 在 这 样 创 建 一 个 独立 的 程序 ， 文 件 中 的 一 
个 类 必须 与 文件 同名 〈 如 果 没 这 样 做 ， 编 译 器 会 及 时 作出 反应 ) 。 类 里 
必须 包含 一 个 名 为 main() 的 方法 ， 形 式 如 下 : 











public static void main(String[] args) { 

其 中 ， 关 键 字 “public”* 间 味 着 方法 可 由 外 部 世界 调用 (第 5 间 会 详细 解 
释 ) 。main0 的 自 变量 是 包含 了 String 对 象 的 一 个 数组 。args 不 会 在 本 程 
序 中 用 到 ， 但 需要 在 这 个 地 方 列 出 ， 因 为 它们 保存 了 在 命令 行 调用 的 上 自 


变量 。 


程序 的 第 一 行 非常 有 趣 : 





System.out.printIn(new Date()); 


请 观察 它 的 自 变 量 : 创建 Date 对 象 唯 一 的 目的 就 是 将 它 的 值 发 送 给 
println()。 一 旦 这 个 语句 执行 完毕 ，Date 束 不 再 需要 。 随 之 而 来 的 “垃圾 
收集 器 ”会 发 现 这 一 情况 ， 并 在 任何 可 能 的 时 候 将 其 回收 。 事 实 上 ,我 
们 没 太 大 的 必要 关心 “清除 ”的 细节 。 


第 二 行 调用 了 System.getProperties()。 若 用 Web 浏 览 器 查看 联机 用 户 文 
档 ， 就 可 知道 getPropertiesO 是 System 类 的 一 个 static 方 法 。 由 于 它 是 “ 静 
态 ” 的 ， 所 以 不 必 创 建 任何 对 象 便 可 调用 该 方法 。 无 论 是 否 存 在 该 类 的 
一 个 对 象 ，static 方 法 随时 都 可 使 用 。 调 用 getProperties() 时 ， 它 会 将 系统 
属性 作为 Properties 类 的 一 个 对 象 生 成 (注意 Properties 是 “属性 ”的 意 

E) 。 随 后 的 的 句柄 保存 在 一 个 名 为 p 的 Properties 句 柄 里 。 在 第 三 行 ， 
大 家 可 看 到 Properties 对 象 有 一 个 名 为 list( 的 方法 ， 它 将 自己 的 全 部 内 容 
都 发 给 一 个 我 们 作为 目 变 量 传递 的 PrintStream 对 象 。 


main() 的 第 四 和 第 六 行 是 典型 的 打印 语句 。 注 意 为 了 打印 多 个 String 值 ， 
用 加 号 (+) 分 隔 它们 即 可 。 然 而 ， 也 要 在 这 里 注意 一 些 奇怪 的 事情 。 
在 String 对 象 中 使 用 时 ， 加 号 并 不 代表 真正 的 “ 相 加 ”。 处 理 字 串 时 ， 我 
们 通常 不 必 考 虑 “4” 的 任何 特殊 含义 。 但 是 ，Java 的 String 类 要 受 一 种 名 
为 < 运算 符 过 载 > 的 机 制 的 制约 。 也 就 是 说 ， 只 有 在 随同 String 对 象 使 用 
时 ， 加 号 才 会 产生 与 其 他 任何 地 方 不 同 的 表现 。 对 于 字 串 ， 它 的 意思 
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但 事情 到 此 并 未 结束 。 请 观察 下 述 语句 : 
System.out.println("Total Memory =" 

+ rt.totalMemory() 

+" Free Memory =" 

+ rt.freeMemory()); 


其 中 ，totalMemory0O 和 freeMemory0 返 回 的 是 数值 ， 并 非 String 对 象 。 如 
果 将 一 个 数值 “加 ?到 一 个 字 串 身上 ， 会 发 生 什么 情况 呢 ? 同 我 们 一 样 ， 
编译 器 也 会 意识 到 这 个 问题 ， 并 魔术 般 地 调用 一 个 方法 ， 将 那个 数值 
Gint，float 等 等 ) 转换 成 字 串 。 经 这 样 处 理 后 ， 它 们 当然 能 利用 加 

号 “加 ”到 一 起 。 这 种 “自动 类 型 转换 ” 亦 划 入 “运算 符 过 载 ? 处 理 的 范畴 。 


许多 Java 昔 作者 在 热烈 地 辩论 “运算 符 过 载 ”〈C++ 的 一 项 特性 ) 是 否 
用 。 目 前 就 是 反对 它 的 一 个 好 例子 ! 然而 ， 这 最 多 只 能 算 编译 器 〈 程 
Fe) 的 问题 ， 而 且 只 是 对 String 对 象 而 言 。 对 于 目 己 编写 的 任何 源 代 
码 ， 都 不 可 能 使 运算 符 “ 过 载 ”。 


通过 为 Runtime 类 调用 getRuntime0) 方 法 ，main() 的 第 五 行 创建 了 一 个 
Runtime 对 象 。 返 回 的 则 是 指 癌 一 个 Runtime 对 象 的 句 顶 。 而 且 ， 我 们 不 
必 关 心 它 是 一 个 静态 对 象 ， 还 是 由 new 命 令 创建 的 一 个 对 象 。 这 是 由 于 
我 们 不 必 为 清除 工作 负责 ， 可 以 大 模 大 样 地 使 用 对 象 。 正 如 显示 的 那 
样 ，Runtime 可 告诉 我 们 与 内 存 使 用 有 关 的 信息 。 











2.8 VE AEA KA SCS 


Java 里 有 两 种 类 型 的 注释 。 第 一 种 是 传统 的 、C 语 言 风格 的 注释 ， 是 从 
C++ 继承 而 来 的 。 这 些 注释 用 一 个 %*" 起 头 ， 随 后 是 注释 内 容 ， 并 可 路 
越 多 行 ， 最 后 用 一 个 “*/" 结 束 。 注 意 许 多 程序 员 在 连续 注释 内 容 的 每 一 
行 都 用 一 个 “*" 开 头 ， 所 以 经 常 能 看 到 象 下 面 这 样 的 内 容 : 


/+ 这 是 





* 一 段 注释 
* 它 跨 越 了 多 个 行 
*/ 


但 请 记 住 ， 进 行 编译 时 ，* 和 */ 之 间 的 所 有 东西 都 会 被 忽略 ， 所 以 上 述 
注释 与 下 面 这 上 段 注释 并 没有 什么 不 同 : 

上 这 是 一 段 注释 ， 

它 跨越 了 多 个 行 */ 

第 二 种 类 型 的 注释 也 起 源 于 C++。 这 种 注释 叫 作 “单行 注释 ”， 以 一 

个 "//" 起 头 ， 表 示 这 一 行 的 所 有 内 容 都 是 注释 。 这 种 类 型 的 注释 更 党 
用 ， 因 为 它 书 写 时 更 方便 。 没 有 必要 在 键盘 上 寻找 "%”， 再 寻找 “*” (只 
需 按 同样 的 键 两 次 ) ， 而 且 不 必 在 注释 结尾 时 加 一 个 结束 标记 。 下 面 便 
是 这 类 注释 的 一 个 例子 : 


// 这 是 一 条 单行 注释 








2.8.1 注释 文档 


对 于 Java 语 言 ， 最 体贴 的 一 项 设计 就 是 它 并 没有 打算 让 人 们 为 了 写 程序 
而 写 程序 一 一 人 们 也 需要 考虑 程序 的 文档 化 问题 。 对 于 程序 的 文档 化 ， 
最 大 的 问题 英 过 于 对 文档 的 维护 。 辱 文档 与 代码 分 离 ， 那 么 每 次 改变 代 
码 后 都 要 改变 文 要 ， 这 无 疑 会 变 成 相当 厅 烦 的 一 件 事 情 。 解 决 的 方法 看 
起 来 似乎 很 简单 : 将 代码 同文 档 “ 链 接 ?” 起 来 。 为 达到 这 个 目的 ， 最 简单 




















的 方法 是 将 所 有 内 容 都 置 于 同一 个 文件 。 然 而 ， 为 使 一 切 都 整齐 划一 ， 
还 必须 使 用 一 种 特殊 的 注释 语法 ， 以 便 标 记 出 特殊 的 文档 ; 另外 还 需要 
一 个 工具 ， 用 于 提取 这 些 注释 ， 并 按 有 价值 的 形式 将 其 展现 出 来 。 这 些 
都 是 Java 必 须 做 到 的 。 


用 于 提取 注释 的 工具 叫 作 javadoc。 它 采用 了 部 分 来 自 Java 编 译 器 的 技 
术 ， 碍 找 我 们 置 入 程序 的 特殊 注释 标记 。 它 不 仅 提 取 由 这 些 标记 指示 的 
信息 ， 也 将 毗邻 注释 的 类 名 或 方法 名 提取 出 来 。 这 样 一 来 ， 我 们 就 可 用 
最 轻 的 工作 量 ， 生 成 十 分 专业 的 程序 文档 。 


javadoc 输 出 的 是 一 个 HTML 文 件 ， 可 用 自己 的 Web 浏 览 串 查看。 该 工具 
允许 我 们 创建 和 管理 单个 源 文件 ， 并 生动 生成 有 用 的 文档 。 由 于 有 了 
jvadoc， 所 以 我 们 能 够 用 标准 的 方法 创建 文档 。 而 且 由 于 它 非常 方便 ， 
所 以 我 们 能 轻松 获得 所 有 Java 库 的 文档 。 


2.8.2 具体 语法 


所 有 javadoc 命 令 都 只 能 出 现 于 “/**” 注 释 中 。 但 和 平常 一 样 ， 注 释 结 
于 一 个 “*/”。 主 要 通过 两 种 方式 来 使 用 javadoc: HRAAYHTML, AE 
用 “文档 标记 ”。 其 中 , “文档 标记 ”(Doc tags) 是 一 些 以 “@” 开 头 的 命 
令 ， 置 于 注释 行 的 起 始 处 〈 但 前 导 的 “*” 会 被 忽略 ) 。 

有 三 种 类 型 的 注释 文档 ， 它 们 对 应 于 位 于 注释 后 面 的 元 素 : 类 、 变 量 或 
者 方法 。 也 就 是 说 ， 一 个 类 注释 正好 位 于 一 个 类 定义 之 前 ; 变量 注释 正 
好 位 于 变量 定义 之 前 ; 而 一 个 方法 定义 正好 位 于 一 个 方法 定义 的 前 面 。 
如 下 面 这 个 简单 的 例子 所 示 : 


po 一 个 类 注释 岂 




















public class docTest { 
[ee 一 个 变量 注释 */ 
public int i; 

Pe AS TERE */ 


public void fQ) {} 


} 


注意 javadoc 只 能 为 public〈 公 共 ) 和 protected 〈 受 保护 ) 成 员 处 理 注释 

X. “private MAWA) 和 “友好 ”( 详 见 5 章 ) 成 员 的 注释 会 被 忽略 ， 我 
们 看 不 到 任何 输出 《也 可 以 用 -private 标 记 包 括 private 成 员 ) 。 这 样 做 是 
有 道理 的 ， 因 为 只 有 public 和 protected 成 员 才 可 在 文件 之 外 使 用 ， 这 是 

客户 程序 员 的 希望 。 然 而 ， 所 有 类 注释 都 会 包含 到 得 出 结果 里 。 

上 述 代 码 的 输出 是 一 个 HTML 文 件 ， 它 与 其 他 Java 文 档 具 有 相同 的 标准 
格式 。 因 此 ， 用 户 会 非常 熟悉 这 种 格式 ， 可 在 您 设计 的 类 中 方便 地 “ 漫 

游 "。 设 计 程 序 时 ， 请 务必 考虑 输 入 上 述 代 码 ， 用 javadoc 处 理 一 下 ， 观 
看 最 终 HTML 文 件 的 效果 如 何 。 

2.8.3 #kK AHTML 


javadoc 将 HTML 命令 传递 给 最 终生 成 的 HTML 文 档 。 这 便 使 我 们 能 够 充 
分 利用 HIML 的 巨大 威力 。 当 然 ， 我 们 的 最 终 动 机 是 格式 化 代码 ， 不 是 
为 了 哗众取宠 。 下 面 列 出 一 个 例子 : 


/ 米 米 














* <pre> 

* System.out.printIn(new Date()); 
* </pre> 

gi 


亦 可 象 在 其 他 Web 文 档 里 那样 运用 HIML， 对 普通 文本 进行 格式 化 ， 使 
其 更 具 条 理 、 更 加 美观 : 


/六 六 
* 你 <em> 甚 至 </em> 可 以 插入 一 个 列表 : 
* <ol> 


* <li> 项 目 一 


* <li> MHZ 

* <li> A= 

* </ol> 

a 

注意 在 文档 注释 中 ， 位 于 一 行 最 开头 的 星 写 会 被 javadoc 于 弃 。 同 时 丢弃 

的 还 有 前 导 空 格 。javadoc 会 对 所 有 内 容 进行 格式 化 ， 使 其 与 标准 的 文档 

外 观 相符 。 不 要 将 <h1> 或 <hr> 这 样 的 标题 当 作 敬 入 HTML 使 用 ， 因 为 

javadoc 会 插入 目 己 的 标题 ， 我 们 给 出 的 标题 会 与 之 冲撞 。 

所 有 类 型 的 注释 文档 一 类、 变量 和 方法 一 一 都 支持 其 入 HTML 。 

2.8.4 @see: 引用 其 他 类 

所 有 三 种 类 型 的 注释 文档 都 可 包含 @see 标 记 ， 它 允许 我 们 引用 其 他 类 里 

的 文档 。 对 于 这 个 标记 ，javadoc 会 生成 相应 的 HTML， 将 其 直接 链接 到 

其 他 文档 。 格 式 如 下 : 

@see 类 名 

@see 完整 类 名 

@see 完整 类 名 # 方 法 名 

每 一 格式 都 会 在 生成 的 文档 里 目 动 加 入 一 个 超 链接 的 “See Also”( 参 

人 
BM o 

2.8.5 类 文档 标记 


随同 藤 入 HIML 和 人 @see 引 用 ， 关 文档 还 可 以 包括 用 于 版 本 信息 以 及 作者 
姓名 的 标记 。 类 文档 亦 可 用 于 “接口 ?目的 〈 本 书后 面 会 详细 解释 ) 。 





1. @version 


格式 如 下 : 


@version 版 本 信息 


其 中 ,“ 版 本 信息 ”代表 任何 适合 作为 版 本 说 明 的 资料 。 厂 在 javadoc 命 令 
行使 用 了 “-version” 标 记 ， 就 会 从 生成 的 HTML 文 档 里 提取 出 版 本 信息 。 


2. @author 

格式 如 下 : 

@author 作者 信息 

其 中 ,“ 作 者 信息 ”包括 您 的 姓名 、 电 子 函 件 地 址 或 者 其 他 任何 适宜 的 资 
料 。 若 在 javadoc 命 令 行 使 用 了 “-author”* 标 记 ， 就 会 专门 从 生成 的 HTML 
文档 里 提取 出 作者 信息 。 


可 为 一 系列 作者 使 用 多 个 这 样 的 标记 ， 但 它们 必须 连续 放置 。 全 部 作者 
信息 会 一 起 存 入 最 终 HTML 代 码 的 单独 一 个 段落 里 。 


2.8.6 变量 文档 标记 
变量 文档 只 能 包括 嵌入 的 HIML 以 及 @see 引 用 。 
2.8.7 方法 文档 标记 


除 藤 入 HTML 和 @see 引 用 之 外 ， 方 法 还 允许 使 用 针对 参数 、 返 回 值 以 及 
违例 的 文档 标记 。 








1. @param 

格式 如 下 : 

@param 参数 名 说 明 

其 中 , “参数 名 ?是 指 参数 列表 内 的 标识 符 ， 而 说明” 代表 一 些 可 延续 到 
后 续 行 内 的 说 明文 字 。 一 旦 遇 到 一 个 新 文档 标记 ， 束 认为 前 一 个 说 明 结 
束 。 可 使 用 任意 数量 的 说 明 ， 每 个 参数 一 个 。 


2. @retum 


格式 如 下 : 

@return 说 明 

其 中 ,，“ 说 明 ” 是 指 返 回 值 的 含义 。 它 可 延续 到 后 面 的 行内 。 

3. @exception 

ARE” (Exception) 的 详细 情况 ， 我 们 会 在 第 9 章 讲 述 。 简 言 之 ， 
它们 是 一 些 特 殊 的 对 象 ， 知 某 个 方法 失败 ， 束 可 将 它们 “ 扔 出 ”对象 。 调 
用 一 个 方法 时 ， 尽 管 只 有 一 个 违例 对 象 出 现 ， 但 一 些 特殊 的 方法 也 许 能 
产生 任意 数量 的 、 不 同类 型 的 违例 。 所 有 这 些 违例 都 需要 说 明 。 所 以 ， 
违例 标记 的 格式 如 下 : 

@exception 完整 类 名 说 明 

其 中 ,“ 完 整 类 名 ”明确 指定 了 一 个 违例 类 的 名 字 ， 它 是 在 其 他 某 个 地 方 
定义 好 的 。 而 “说 明 ”( 同 样 可 以 延续 到 下 面 的 行 ) 告诉 我 们 为 什么 这 种 
特殊 类 型 的 违例 会 在 方法 调用 中 出 现 。 

4. @deprecated 

这 是 Java 1.1 的 新 特性 。 该 标记 用 于 指出 一 些 旧 功能 已 由 改进 过 的 新 功 
能 取代 。 该 标记 的 作用 是 建议 用 户 不 必 再 使 用 一 种 特定 的 功能 ， 因 为 未 
来 改 厂 时 可 能 据 弃 这 一 功能 。 知 将 一 个 方法 标记 为 @deprecated， 则 使 
用 该 方法 时 会 收 到 编译 器 的 警告 。 

2.8.8 文档 示例 

下 面 还 是 我 们 的 第 一 个 Java 程 序 ， 只 不 过 已 加 入 了 完整 的 文档 注释 : 


92 页 程序 














第 一 行 : 
//: Property.java 


采用 了 我 目 己 的 方法 : 将 一 个 “:” 作 为 特殊 的 记 写 ， 指 出 这 是 包含 了 源 文 
件 名 字 的 一 个 注释 行 。 最 后 一 行 也 用 这 样 的 一 条 注释 结尾 ， 它 标志 着 源 





代码 清单 的 结束 。 这 样 一 来 ， 可 将 代码 从 本 书 的 正文 中 方便 地 提取 出 
来 ， 并 用 一 个 编译 句 检 查 。 这 方面 的 细节 在 第 17 章 讲述 。 


2.9 编码 样式 

一 个 非 正 式 的 Java 编 程 标准 是 大 写 一 个 类 名 的 首 字 母 。 知 类 名 由 几 个 单 
词 构成 ， 那 么 把 它们 紧 靠 到 一 起 〈 也 就 是 说 ， 不 要 用 下 划 线 来 分 隔 名 
字 ) 。 此 外 ， 每 个 能 入 单词 的 首 字 母 都 采用 大 写 形式 。 例 如 : 


class AllTheColorsOfTheRainbow { //...} 


对 于 其 他 几乎 所 有 内 容 : 方法 、 字 段 〈 成 员 变 量 ) 以 及 对 象 句柄 名 称 ， 
人 
H: 











class AllTheColorsOfTheRainbow { 
int anIntegerRepresentingColors; 


void changeTheHueOfTheColor(int newHue) { 


当然 ， 要 注意 用 户 也 必须 键入 所 有 这 些 长 名 字 ， 而 且 不 能 输 错 。 





2.10 总 结 


通过 本 章 的 学 习 ， 大 家 已 接触 了 足够 多 的 Java 编 程 知识 ， 已 知道 如 何 自 
行 编写 一 个 简单 的 程序 。 此 外 ， 对 语言 的 总 体 情 况 以 及 一 些 基 本 思想 也 
有 了 一 定 程度 的 认识 。 然 而 ， 本 章 所 有 例子 的 模式 都 是 单线 形式 的 “这 
样 做 ， 再 那样 做 ， 然 后 再 做 男 一 些 事情 ”。 如 果 想 让 程序 作出 一 项 选 
择 ， 又 该 如 何 设计 呢 ? 例如 , “假如 这 样 做 的 结果 是 红色 ， 就 那样 做 ; 
如 果 不 是 ， 束 做 另 一 些 事情 ?”。 对 于 这 种 基本 的 编程 方法 ， 下 一 章 会 详 
细 说 明 在 Java 里 是 如 何 实 现 的 。 








2.11 练习 


(1) 参照 本 章 的 第 一 个 例子 ， 创 建 一 个 “*Hello，World” 程 序 ， 在 屏幕 上 简 
单 地 显示 这 人 句 话 。 注 意 在 自己 的 类 里 只 需 一 个 方法 (“main” 方 法 会 在 程 
序 启动 时 执行 )。 记 住 要 把 它 设 为 static 形 式 ， 并 置 入 自 变 量 列表 一 一 即 
使 根本 不 会 用 到 这 个 列表 。 用 javac 编 译 这 个 程序 ， 再 用 java 运 行 它 。 


(2) 写 一 个 程序 ， 打 印 出 从 命令 行 获取 的 三 个 自 变 量 。 


(3) ” 找 出 Propertyjava 第 二 个 版 本 的 代码 ， 这 是 一 个 简单 的 注释 文档 示 
例 。 请 对 文件 执行 javadoc， 并 在 自己 的 Web 浏 览 器 里 观看 结 


(4) 以 练习 (1) 的 程序 为 基础 ， 向 其 中 加 入 注释 文档 。 利 用 javadoc， 将 这 
个 注释 文档 提取 为 一 个 HTML 文 件 ， 并 用 Web 浏 览 器 观看 。 








第 3 草 控制 程序 流程 


“就 象 任何 有 感知 的 生物 一 样 ， 程 序 必须 能 操纵 上 自己 的 世界 ， 在 执行 过 
程 中 作出 判断 与 选择 。” 


在 Java 里 ， 我 们 利用 运算 符 操 纵 对 象 和 数据 ， 并 用 执行 控制 语句 作出 选 
择 。Java 是 建立 在 C++ 基础 上 的 ， 所 以 对 C 和 C++ 程序 员 来 说 ， 对 Java 这 
方面 的 大 多 数 语句 和 运算 符 都 应 是 非 党 熟悉 的 。 当 然 ，Java 也 进行 了 目 
己 的 一 些 改 进 与 简化 工作 。 





3.1 使 用 Java 运 算 符 


运算 符 以 一 个 或 多 个 目 变 量 为 基础 ， 可 生成 一 个 新 值 。 目 变量 采用 与 原 
台 方 法 调用 不 同 的 一 种 形式 ， 但 效果 是 相同 的 。 根 据 以 前 写 程序 的 经 
验 ， 运 算 符 的 常规 概念 应 该 不 难 理解 。 


加 号 〈+) 、 减 号 和 负 号 〈-) 、 乘 号 (*) 、 除 号 (/) 以 及 等 号 (=) 
的 用 法 与 其 他 所 有 编程 语言 都 是 类 似 的 。 


所 有 运算 符 都 能 根据 自己 的 运算 对 象 生成 一 个 值 。 除 此 以 外 ， 一 个 运算 
符 可 改变 运算 对 象 的 值 ， 这 叫 作 “ 副 作用 ”(Side Effect) 。 运 算 符 最 党 
见 的 用 途 就 是 修改 自己 的 运算 对 象 ， 从 而 产生 副作用 。 但 要 注意 生成 的 
值 亦 可 由 没有 副作用 的 运算 符 生 成 。 


几乎 所 有 运算 符 都 只 能 操作 * 主 类 型 ”(Primitives) 。 唯 一 的 例外 

是 “=”“==” 和 “1=>， 它 们 能 操作 所 有 对 象 〈 也 是 对 象 易 令 人 混 消 的 一 
个 地 方 )。 除 此 以 外 ，String 类 支持 “+” 和 “+=”。 

3.1.1 优先 级 

运算 符 的 优先 级 决定 了 存在 多 个 运算 符 时 一 个 表达 式 各 部 分 的 计算 顺 
序 。Java 对 计算 顺序 作出 了 特别 的 规定 。 其 中 ， 最 简单 的 规则 就 是 乘法 
和 除法 在 加 法 和 减法 之 前 完成 。 程 序 员 经 党 都 会 在 记 其 他 优先 级 规则 ， 
所 以 应 该 用 括号 明确 规定 计算 顺序 。 例 如 : 

A=X+Y-2/2+Z， 

为 上 述 表达 式 加 上 括号 后 ， 就 有 了 一 个 不 同 的 含义 。 
A=X+(Y-2)/(2+Z); 

3.1.2 赋值 

赋值 是 用 等 号 运算 符 (=|) 进行 的 。 它 的 意思 是 “取得 右边 的 值 ， 把 它 复 
制 到 左边 ?。 右 边 的 值 可 以 是 任何 和 常数、 变量 或 者 表达 式 ， 只 要 能 产生 


一 个 值 残 行 。 但 左边 的 值 必 须 是 一 个 明确 的 、 已 命名 的 变量 。 也 就 古 
说 ， 它 必须 有 一 个 物理 性 的 空间 来 保存 石 边 的 值 。 举 个 例子 来 说 ， 可 将 









































一 个 常数 赋 给 一 个 变量 (A=4;) ， 但 不 可 将 任何 东西 赋 给 一 个 常数 〔 比 
如 不 能 4=A) 。 


对 主 数据 类 型 的 赋值 是 非常 直接 的 。 由 于 主 类 型 容纳 了 实际 的 值 ， 而 且 
并 非 指 加 一 个 对 象 的 句柄 ， 所 以 在 为 其 赋值 的 时 候 ， 可 将 来 自 一 个 地 方 
的 内 容 复 制 到 男 一 个 地 方 。 例 如 ， 假 设 为 主 类 型 使 用 “A=B”， 那 么 B 处 
的 内 容 束 复制 到 A。 吞 接着 义 修改 了 A， 那 么 B 根 本 不 会 受 这 种 修改 的 影 
啊 。 作 为 一 名 程序 员 ， 这 应 成 为 自己 的 常识 。 


但 在 为 对 象 “ 赋 值 * 的 时 候 ， 情 况 却 及 生 了 变化 。 对 一 个 对 象 进行 操作 
时 ， 我 们 真正 操作 的 是 它 的 句柄 。 所 以 倘 硅 “从 一 个 对 象 到 男 一 个 对 
象 ” 赋 值 ， 实 际 就 是 将 句柄 从 一 个 地 方 复制 到 男 一 个 地 方 。 这 意味 着 假 
知 为 对 象 使 用 “C=D”， 那 么 C 和 DD 最 终 都 会 指 疝 最 初 只 有 DD 才 指 向 的 那个 
对 象 。 下 面 这 个 例子 将 向 大 家 阐 示 这 一 后。 


这 里 有 一 些 题 外 话 。 在 后 面 ， 大 家 在 代码 示例 里 看 到 的 第 一 个 语句 将 
是 “package 03” 使 用 的 “package” 语 句 ， 它 代表 本 书 第 3 章 。 本 书 每 一 章 的 
第 一 个 代码 清单 都 会 包含 象 这 样 的 一 个 "package”(〈 封 装 、 打 包 、 包 囊 ) 
语句 ， 它 的 作用 是 为 那 一 章 剩 余 的 代码 建立 章节 编号 。 在 第 17 章 ， 大 家 
会 看 到 第 3 章 的 所 有 代码 清单 〈 除 那些 有 不 同 封装 名 称 的 以 外 ) 都 会 目 
动 置 入 一 个 名 为 c03 的 子 目 录 里 ; 第 4 章 的 代码 置 入 c04; 以 此 类 推 。 所 
有 这 些 都 是 通过 第 17 章 展示 的 CodePackage.java 程 序 实现 的 ;“ 封 装 ” 的 
基本 概念 会 在 第 5 章 进 行 详尽 的 解释 。 就 目前 来 说 ， 大 家 只 需 记 住 
‘package 03” 这 样 的 形式 只 是 用 于 为 某 一 章 的 代码 清单 建立 相应 的 子 





























为 运行 程序 ， ee 
目录 〈 那 个 目录 里 包含 了 c02，c03c，c04 等 等 子 目 录 


对 于 Java 后 续 的 版 本 〈1.1.4 和 更 高 版 本 ) ， es ee 
名 封装 到 一 个 文件 里 ， 那 么 必须 在 程序 名 前 面 指定 完整 的 包 囊 名 称 ， 
则 不 能 运行 程序 。 在 这 种 情况 下 ， 命 令 行 是 : 








java c03.Assignment 
运行 位 于 一 个 " 包 衷 ”里 的 程序 时 ， 随 时 都 要 注意 这 方面 的 问题 。 
下 面 是 例子 : 


//: Assignment.java 


// Assignment with objects is a bit tricky 


package c03; 


Class Number { 


} 


int 1; 


public class Assignment { 


Number n1 


Number n2 


S 
H 
H 
lI 


niet = 27; 


public static 


.println("1: 


.println("2: 


System.out. 


void main(String[] args) { 


new Number (); 


new Number (); 


" + n2.i); 


" + n2.i); 


printin("3: 


" + n2.i); 


n1. 


n1. 


n1. 


i: 


1: 


i: 


" + n.i + 


"+ n1.i + 


" + n.i + 


Number 类 非常 简单 ， 它 的 两 个 实例 aMn) 是 在 main(0) 里 创建 的 。 
个 Number 中 的 i 值 都 赋予 了 一 个 不 同 的 值 。 随 后 ， 将 n2 赋 给 n1， 而 且 nl 
发 生 改变 。 在 许多 程序 设计 语言 中 ， 我 们 都 希望 n1 和 n2 任 何 时 候 都 相互 
独立 。 但 由 于 我 们 已 趴 予 了 一 个 句柄 ， 所 以 下 面 才 是 真实 的 输出 : 


1: n1.i: 9, n2.i: 47 





2: n1.i: 47, n2.i: 47 
3: n1.i: 27, n2.i: 27 


看 来 改变 n1 的 同时 也 改变 了 n2! 这 是 由 于 无 论 n1 还 是 n2 都 包含 了 相同 的 
句柄 ， 它 指向 相同 的 对 象 〈 最 初 的 句柄 位 于 n1 内 部 ， 指 向 容纳 了 值 9 的 
一 个 对 象 。 在 赋值 过 程 中 ， 那 个 句柄 实际 已 经 丢失 ， 它 的 对 象 会 由 “ 垃 
圾 收集 器 ? 目 动 清除 ) 。 


这 种 特殊 的 现象 通常 也 叫 作 “别名 ”， 是 Java 操 作对 象 的 一 种 基本 方式 。 
但 假 大 不 愿意 在 这 种 情况 下 出 现 别名 ， 叉 该 怎么 操作 呢 ? 可 放弃 赋值 ， 
并 写 入 下 述 代码 : 











nl. = n2.1; 


这 样 便 可 保留 两 个 独立 的 对 象 ， 而 不 是 将 n1 和 n2 绑 定 到 相同 的 对 象 。 但 
您 很 快 融会 意识 到 ， 这 样 做 会 使 对 象 内 部 的 字段 处 理发 生 混乱 ， 并 与 标 
准 的 面向 对 象 设计 准则 相 怪 。 由 于 这 并 非 一 个 简单 的 话题 ， 所 以 留待 第 
12 章 详细 论述 ， 那 一 章 是 专门 讨论 别名 的 。 其 时 ， 大 家 也 会 注意 到 对 象 
的 赋值 会 产生 一 些 令 人 震惊 的 效果 。 

1. 方法 调用 中 的 别名 处 理 

将 一 个 对 象 传递 到 方法 内 部 时 ， 也 会 产生 别名 现象 。 


//: PassObject.java 





// Passing objects to methods can be a bit tricky 
class Letter { 


char c; 


} 


public class PassObject { 


static void f(Letter y) { 


public static void main(String[] args) { 


Letter x = new Letter(); 


Kiet als 
System.out.println("1: x.c: " + X.C); 
f(x); 
System.out.println("2: x.c: " + x.c); 
} 
} ///:~ 


在 许多 程序 设计 语言 中 ，fO0 方 法 表面 上 似乎 要 在 方法 的 作用 域内 制作 上 自 
己 的 目 变 量 Letter y 的 一 个 副本 。 但 同样 地 ， 实 际 传递 的 是 一 个 句柄 。 所 
以 下 面 这 个 程序 行 : 

y.c = Z; 


实际 改变 的 是 f0 之 外 的 对 象 。 输 出 结果 如 下 : 





1: x.c:a 


2: X.C: Z 


se 
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所 有 答案 ， 但 从 现在 开始 就 应 加 以 重视 ， 以 便 提早 发 现 它 的 缺点 。 











3.1.3 算术 运算 符 

Java 的 基本 算术 运算 符 与 其 他 大 多 数 程序 设计 语言 是 相同 的 。 其 中 包括 
加 号 (+) 、 减 号 (-) 、 除 号 (/) 、 乘 号 (*) 以 及 模 数 (9%6， 从 整数 
除法 中 获得 余数 ) 。 整 数 除法 会 直接 砍 掉 小 数 ， 而 不 是 进位 。 

Java 也 用 一 种 简写 形式 进行 运算 ， 并 同时 进行 赋值 操作 。 这 是 由 等 写 前 
的 一 个 运算 符 标 记 的 ， 而 且 对 于 语言 中 的 所 有 运算 符 都 是 固定 的 。 例 

如 ， 为 了 将 4 加 到 变量 x， 并 将 结果 赋 给 x， 可 用 : x+=4。 

下 面 这 个 例子 展示 了 算术 运算 符 的 各 种 用 法 : 


//: MathOps.java 




















// Demonstrates the mathematical operators 
import java.util.*; 
public class MathOps { 
// Create a shorthand to save typing: 
static void prt(String s) { 
System.out.println(s); 
} 
// shorthand to print a string and an int: 
static void pInt(String s, int i) { 
prea? e a); 
} 
// shorthand to print a string and a float: 
static void pFlt(String s, float f) { 


prt(s + "= "+ f); 


} 
public static void main(String[] args) 
// Create a random number generator, 
// seeds with current time by default: 
Random rand = new Random(); 
int i, j, k; 


// ‘%' limits maximum value to 99: 


j rand.nextInt() % 100; 


k = rand.nextInt() % 100; 


pInt("j",j); piInt("k",k); 


i 


lI 
E 
+ 
x 


pInt("j + k", i); 

i= j - k; pInt("j - k", i); 

i =k / j; pInt("k / j", i); 

i =k * j; pIīInt("k * j", i); 

i=k% Jj; pInt("k % j", i); 

j %= k; pInt("j %= k", j); 

// Floating-point number tests: 
float u,v,w; // applies to doubles, too 


v = rand.nextFloat(); 


W rand.nextFloat(); 
pFlt("v", v); pF1lt("w", w); 


u =v + w; pFlt("v + w", u); 


C 
lI 


v - w; pFlt("v - w", u); 


u =v * w; pFlt("v * w", u); 

u = v / w; pFlt("v / w", u); 

// the following also works for 
// char, byte, short, int, long, 
// and double: 

u += v; pFlt("u += v", u); 

u -= v; pFlt("u -= v", u); 

u *= v; pFlt("u *= v", u); 


u /= v; pFlt("u /= v", u); 


} ///:~ 





我 们 注意 到 的 第 一 件 事情 就 是 用 于 打印 (显示 ) 的 一 些 快捷 方法 : prt() 
方法 打印 一 个 String; pInt0 先 打印 一 个 String， 再 打印 一 个 int， 而 pFlt() 
先 打 印 一 个 String， 再 打印 一 个 float。 当 然 ， 它 们 最 终 都 要 用 
System.out.println(O) 结 尾 。 


为 生成 数字 ， 程 序 首 先 会 创建 一 个 Random (EID 对 象 。 由 于 上 自 变 量 
是 在 创建 过 程 中 传递 的 ， 所 以 Java 将 当前 时 间作 为 一 个 “种 子 值 "”， 由 随 
机 数 生 成 器 利用 。 通 过 Random 对 象 ， 程 序 可 生成 许多 不 同类 型 的 随机 
数字 。 做 法 很 简单 ， 只 需 调 用 不 同 的 方法 即 可 : nextInt()，nextLong()， 
nextFloat() 或 者 nextDouble()。 


若 随同 随机 数 生 成 器 的 结果 使 用 ， 模 数 运算 符 〈%) 可 将 结果 限制 到 运 
算 对 象 减 1 的 上 限 〈 本 例 是 99) 之 下 。 


1. 一 元 加 、 减 运算 符 


一 元 减 号 〈-) 和 一 元 加 号 (+) 与 二 元 加 号 和 减 号 都 是 相同 的 运算 符 。 
根据 表达 式 的 书写 形式 ， 编 译 右 会 目 动 判断 使 用 哪 一 种 。 例 如 下 述 语 





它 的 含义 是 显然 的 。 编 译 器 能 正确 识别 下 述 语句 : 
X=a*-b; 

但 读者 会 被 搞 糊 深 ， 所 以 最 好 更 明确 地 写成 : 

x = a * (-b); 


一 元 减 号 得 到 的 运算 对 象 的 负 值 。 一 元 加 号 的 含义 与 一 元 减 号 相反 ， 虽 
然 它 实际 并 不 做 任何 事情 。 


3.1.4 目 动 递增 和 递减 


和 CC 类似，Java 提 供 了 丰富 的 快捷 运算 方式 。 这 些 快捷 运算 可 使 代码 更 
清爽 ， 更 易 录入 ， 也 更 易 读者 办 读 。 


两 种 很 不 错 的 快捷 运算 方式 是 递增 和 递减 运算 符 《〈 常 称 作 <“ 自动 递 

增 ” 和 * 目 动 递减 ?运算 符 ) 。 其 中 ， 递 减 运算 符 是 二 -”， 意 为 “减少 一 个 
单位 ?>， 递 增 运算 符 是 “++”， 意 为 “增加 一 个 单位 >。 举 个 例子 来 说 ， 假 
设 A 是 一 个 int (整数) 值 ， 则 表达 式 ++A 就 等 价 于 (A = A + 1) 。 递 增 
和 递减 运算 符 结 果 生 成 的 是 变量 的 值 。 


对 每 种 类 型 的 运算 符 ， 都 有 两 个 版 本 可 供 选 用 ;， 通 季 将 其 称 为 "前 绥 
版 ?和 "后 绥 厂 ”。“ 前 递增 ?表示 ++ 运 算 符 位 于 变量 或 表达 式 的 前 面 ; 

而 “后 递增 ?表示 ++ 运 算 符 位 于 变量 或 表达 式 的 后 面 。 类 似 地 , “前 递 
减 ” 意 味 着 -- 运 算 符 位 于 变量 或 表达 式 的 前 面 ， 而 “后 递减 ”意味 着 -- 运 算 
符 位 于 变量 或 表达 式 的 后 面 。 对 于 前 递增 和 前 递减 〈 如 ++A 或 --A) ， 
会 先 执行 运算 ， 再 生成 值 。 而 对 于 后 递增 和 后 递减 〈 如 A++ 或 A--) ， 
会 先生 成 值 ， 再 执行 运算 。 下 面 是 一 个 例子 : 


//: AutoInc.java 





























// Demonstrates the ++ and -- operators 


public class AutoInc { 


public static void main(String[] args) { 


int i = 1; 


prt("i : 


prt("++i : 


prt("i++ : 


prt("i : 


prt("--i : 


prt("i-- 
prt("i : 


} 


i); 
+ ++i); // Pre-increment 
+ i++); // Post-increment 
i); 
+ --i); // Pre-decrement 


+ 1--); // Post-decrement 


i); 


static void prt(String s) { 


System.out.println(s); 


} 
NS fief E 


该 程序 的 输出 如 下 : 











从 中 可 以 看 到 ， 对 于 前 级 形式 ， 我 们 在 执行 完 运 算 后 才 得 到 值 。 但 对 于 
后 级 形式 ， 则 是 在 运算 执行 之 前 束 得 到 值 。 它 们 古 唯一 具有 “副作用 ”的 
运算 符 《〈 除 那些 涉及 赋值 的 以 外 ) 。 也 就 是 说 ， 它 们 会 改变 运算 对 象 ， 
而 不 仅仅 是 使 用 自己 的 值 。 


递增 运算 符 正 是 对 “C++” 这 个 名 字 的 一 种 解释 ， 上 暗示 着 “超载 C 的 一 
步 ”。 在 早期 的 一 次 Java 演 讲 中 ，B 记 Joy〈 始 创 人 之 一 ) 声 
称 “Java=C++--”(C 加 加 减 减 ) ， 意 味 着 Java 已 去 除了 C++ 一 些 没 来 由 折 
磨 人 的 地 方 ， 形 成 一 种 更 精简 的 语言 。 正 如 大 家 会 在 这 本 书 中 学 到 的 那 
样 ，Java 的 许多 地 方 都 得 到 了 简化 ， 所 以 Java 的 学 习 比 C++ 更 容易 。 
3.1.5 关系 运算 符 

关系 运算 符 生 成 的 是 一 个 “布尔 ”(Boolean) 结果 。 它 们 评价 的 是 运算 
对 象 值 之 间 的 关系 。 香 关系 是 真实 的 ， 关 系 表 达 式 会 生成 true〈 真 ) ; 
若 关 系 不 真实 ， 则 生成 false( 假 )。 关 系 运 算 符 包括 小 于 (<) 、 大 于 
(> 水 于 或 等 于 大 于 或 等 于 C=) . SF CH=) 以 及 不 等 


F =) 。 等 于 和 不 等 于 适用 于 所 有 内 建 的 数据 类 型 ， 但 其 他 比较 不 适 
用 于 boolean 类 型 。 


1. 检查 对 象 是 人 否 相 等 


关系 运算 符 == 和 != 也 适用 于 所 有 对 象 ， 但 它们 的 含义 通常 会 使 初 涉 Java 
领域 的 人 找 不 到 北 。 下 面 是 一 个 例子 : 


//: Equivalence.java 








public class Equivalence { 
public static void main(String[] args) { 
Integer ni = new Integer(47); 


Integer n2 = new Integer(47); 


System.out.printin(n1 == n2); 
System.out.printin(n1 != n2); 


} 
} ///:~ 





其 中 ， 表 达 式 System.out.printIn(n1 ==”n2) 可 打印 出 内 部 的 布尔 比较 结 
果 。 一 般 人 都 会 认为 输出 结果 肯定 先是 tue， 再 是 false， 因 为 两 个 
Integer 对 象 都 是 相同 的 。 但 尽管 对 象 的 内 容 相 同 ， 句 顶 却 是 不 同 的 ， 而 
== 和 != 比 较 的 正好 就 是 对 象 句柄 。 所 以 输出 结果 实际 上 先是 false， 再 是 
true。 这 自然 会 使 第 一 次 接触 的 人 感到 惊奇 。 

若 想 对 比 两 个 对 象 的 实际 内 容 是 否 相 同 ， 又 该 如 何 操作 呢 ? 此 时 ， 必 须 
使 用 所 有 对 象 都 适用 的 特殊 方法 equals()。 但 这 个 方法 不 适用 于 “ 主 类 
型 >”， 那 些 类 型 直接 使 用 == 和 != 即 可 。 下 面 举例 说 明 如 何 使 用 : 


//: EqualsMethod , java 





public class EqualsMethod { 
public static void main(String[] args) { 
Integer ni = new Integer(47); 
Integer n2 = new Integer(47); 
System.out.println(n1.equals(n2)); 
} 
} ///:~ 


正如 我 们 预计 的 那样 ， 此 时 得 到 的 结果 是 true。 但 事情 并 未 到 此 结束 ! 
假设 您 创建 了 自己 的 类 ， 就 象 下 面 这 样 : 


//: EqualsMethod2.java 


class Value { 
int 1; 
} 
public class EqualsMethod2 { 
public static void main(String[] args) { 
Value vi = new Value(); 
Value v2 = new Value(); 
V1.i = v2.i = 100; 
System.out.println(vi.equals(v2) ); 
} 
} ///:~ 


此 时 的 结果 又 变 回 了 false! 这 是 由 于 equals0 的 默认 行为 是 比较 句柄 。 所 

以 除非 在 自己 的 新 类 中 改变 了 equals 中 ， 否 则 不 可 能 表现 出 我 们 希望 的 
行为 。 不 秆 的 是 ， 要 到 第 7 章 才 会 学 习 如 何 改 变 行为 。 但 要 注意 equals() 
的 这 种 行为 方式 同时 或 许 能 够 避免 一 些 “ 灾 难 ” 性 的 事件 。 


大 多 数 Java 类 库 都 实现 了 equals0， 所 以 它 实际 比较 的 是 对 象 的 内 容 ， 而 
非 它 们 的 句柄 。 
3.1.6 逻辑 运算 符 


逻辑 运算 符 AND (&&) 、OR (||) ARNOT C) 能 生成 一 个 布尔 值 
(true false) 一 一 以 自 变 量 的 逻辑 关系 为 基础 。 下 面 这 个 例子 同 大 家 
展示 了 如 何 使 用 关系 和 逮 辑 运算 符 。 


//: Bool.java 











// Relational and logical operators 


import java.util.*; 


public class Bool { 


public static void main(String[] args) 


Random rand = new Random(); 
int i = rand.nextInt() % 100; 
int j = rand.nextInt() % 100; 
prt("i = " + i); 

PECI hag) 

prt("i > j is " + (i > j)); 
prt("i < j is "+ (i < j)); 
prt("i >= j is " + (i >= j)); 
prt("i <= j is " + (i <= j)); 
prt("i == j is " + (i == j)); 
prt("i != j is " + (i != j)); 


// Treating an int as a boolean is 


// not legal Java 


//! 


//! 


//! 


prt("i && j is "+ (i && j)); 

prt("i || j is "+ (i |] j)); 

prt("!i is " + li); 

prt("(i < 10) && (j < 10) is " 
+ ((i < 10) && (j < 10)) ); 

prt("(i < 10) |] (j < 10) is " 


+ ((i < 10) || (j < 10)) ); 


} 
static void prt(String s) { 
System.out.println(s); 
} 
} ///:~ 


只 可 将 AND，OR 或 NOT 应 用 于 布尔 值 。 与 在 C 及 C++ 中 不 同 ， 不 可 将 一 
个 非 布 尔 值 当 作 布尔 值 在 逻辑 表达 式 中 使 用 。 若 这 样 做 ， 就 会 发 现 尝 试 
失败 ， 并 用 一 个 “/! 标 出 。 然 而 ， 后 续 的 表达 式 利 用 关系 比较 生成 布尔 
值 ， 然 后 对 结果 进行 逻辑 运算 。 


和 输出 列表 看 起 来 过 下 面 这 个 样子 : 
i = 85 
ee. 


i > j is true 

i < j is false 

i >= j is true 

i <= j is false 

i == j is false 

i != j is true 

(i < 10) && (j < 10) is false 


(i < 10) || (j < 10) is true 


注意 大 在 预计 为 String 值 的 地 方 使 用 ， 布 尔 值 会 目 动 转换 成 适当 的 文本 
形式 。 


在 上 述 程序 中 ， 可 将 对 int 的 定义 谷 换 成 除 boolean 以 外 的 其 他 任何 主 数 
据 类 型 。 但 要 注意 ， 对 译 点 数字 的 比较 是 非常 严格 的 。 即 使 一 个 数字 仅 
在 小 数 部 分 与 男 一 个 数字 存在 极 微小 的 差异 ， 仍 然 认 为 它们 是 “不 相 
等 ?的 。 即 使 一 个 数字 只 比 零 大 一 点 点 (例如 2 不 停 地 开平 方 根 ，， 它 仍 
然 属 于 “ 非 零 ” 值 。 


1. 短路 

操作 逻辑 运算 符 时 ， 我 们 会 遇 到 一 种 名 为 “短路 ”的 情况 。 这 意味 着 只 有 
明确 得 出 整个 表达 式 真 或 假 的 结论 ， 才 会 对 表达 式 进行 逻辑 求 值 。 
此 ， 一 个 逻辑 表达 式 的 所 有 部 分 都 有 可 能 不 进行 求 值 : 


//: ShortCircuit.java 














// Demonstrates short-circuiting behavior 
// with logical operators. 
public class ShortCircuit { 
static boolean testi(int val) { 
System.out.printin("test1i(" + val + ")"); 
System.out.printin("result: " + (val < 1)); 
return val < 1; 
} 
static boolean test2(int val) { 
System.out.println("test2(" + val + ")"); 
System.out.printlin("result: " + (val < 2)); 
return val < 2; 
} 


static boolean test3(int val) { 


System.out.printin("test3(" + val + ")"); 
System.out.printin("result: " + (val < 3)); 
return val < 3; 
} 
public static void main(String[] args) { 
if(test1(0) && test2(2) && test3(2)) 
System.out.printin("expression is true"); 
else 
System.out.printin("expression is false"); 
} 
} ///:~ 


每 次 测试 都 会 比较 自 变 量 ， 并 返回 真 或 假 。 它 不 会 显示 与 准备 调用 什么 
有 关 的 资料 。 测 试 在 下 面 这 个 表达 式 中 进行 : 


if(test1(0)) && test2(2) && test3(2)) 


很 自然 地 ， 你 也 许 认 为 所 有 这 三 个 测试 都 会 得 以 执行 。 但 希望 输出 结 
不 至 于 使 你 大 吃 一 惊 : 


if(test1(0) && test2(2) && test3(2)) 


第 一 个 测试 生成 一 个 true 结 果 ， 所 以 表达 式 求 值 会 继续 下 去 。 然 而 ， 第 
二 个 测试 产生 了 一 个 false 结 果 。 由 于 这 意味 着 整个 表达 式 肯 定 为 false， 
所 以 为 什么 还 要 继续 剩余 的 表达 式 呢 ? 这 样 做 只 会 徙 元 无 益 。 事 实 
E, “短路 "一 词 的 由 来 正 种 因 于 此 。 如 果 一 个 逻辑 表达 式 的 所 有 部 分 都 
不 必 执 行 下 去 ， 那 么 潜在 的 性 能 提升 将 是 相当 可 观 的 。 


3.1.7 按 位 运算 符 








按 位 运算 符 人 允许 我 们 操作 一 个 整数 主 数据 类 型 中 的 单个 “比特 ”"， 即 二 进 
ie een aie 
一 人 有 纺 本 。 


按 位 运算 来 源 于 C 语 言 的 低级 操作 。 我 们 经 常 都 要 直接 操纵 硬件 ， 需 要 
频繁 设置 便 件 寄存 器 内 的 二 进 制 位 。Java 的 设计 初衷 是 嵌入 电视 顶 置 盒 
内 ， 所 以 这 种 低级 操作 仍 被 保留 下 来 了 。 然 而 ， 由 于 操作 系统 的 进步 ， 
现在 也 许 不 必 过 于 频繁 地 进行 授 位 运算 。 


若 两 个 输入 位 都 是 1， 则 按 位 AND 运 算 符 (&) 在 输出 位 里 生成 一 个 1; 
否则 生成 0。 若 两 个 输入 位 里 至 少 有 一 个 是 1， 则 按 位 OR 运算 符 (|) 在 
输出 位 里 生成 一 个 1; 只 有 在 两 个 输入 位 都 是 0 的 情况 下 ， 它 才 会 生成 一 
个 0。 若 两 个 输入 位 的 某 一 个 是 1， 但 不 全 都 是 1， 那 么 按 位 XOR O, 
或 ) 在 输出 位 里 生成 一 个 1。 按 位 NOT (~， 也 叫 作 * 非 > 运算 符 ) 属于 一 
元 运算 符 ， 它 只 对 一 个 自 变 量 进行 操作 (其 他 所 有 运算 符 都 是 二 元 运算 
ee ee ee 若 输 入 0， 则 输出 1;， 输入 
1， 则 输出 0。 


按 位 运算 符 和 逻辑 运算 符 都 使 用 了 同样 的 字符 ， 只 古 数 量 不 同 。 因 此 ， 
我 们 能 方便 地 记忆 各 目的 含义 : 由 于 “位 ?是 非常 “小 "的 ， 所 以 按 位 运算 
符 仅 使 用 了 一 个 字符 。 


ERAT SSS (=) 联合 使 用 ， 以 便 合 并 运算 及 赋值 : &=，|= 和 
^= 都 是 合法 的 〈 由 于 ~ 是 一 元 运算 符 ， 所 以 不 可 与 = 联合 使 用 ) 。 


我 们 将 boolean〈 布 尔 ) 类 型 当 作 一 种 “单位 ”或 “ 单 比特 ” 值 对待 ， 所 以 它 
多 少 有 些 独 特 的 地 方 。 我 们 可 执行 按 位 AND，OR 和 XOR， 但 不 能 执行 
按 位 NOT 〈 大 概 是 为 了 避免 与 逻辑 NOT 混 淆 ) 。 对 于 布尔 值 ， 按 位 运算 
符 具 有 与 逻辑 运算 符 相 同 的 效果 ， 只 是 它们 不 会 中 途 “ 短 路 ”。 此 外 ， 针 
对 布尔 值 进行 的 按 位 运算 为 我 们 新 增 了 一 个 XOR 逻 辑 运 算 符 ， 它 并 未 包 
括 在 “逻辑 ”运算 符 的 列表 中 。 在 移 位 表达 式 中 ， 我 们 被 禁止 使 用 布尔 运 
算 ， 原 因 将 在 下 面 解释 。 


3.1.8 移 位 运算 符 
移 位 运算 符 面 向 的 运算 对 象 也 是 二 进 制 的 “位 >。 可 单独 用 它们 处 理 整 数 


类 型 〈 主 类 型 的 一 种 ) 。 堪 移 位 运算 符 〈<<) 能 将 运算 符 左边 的 运算 对 
象 癌 左 移动 运算 符 右 侧 指定 的 位 数 《〈 在 低位 补 0) 。“ 有 符号 ” 右 移 位 运 
































算 符 (>>) 则 将 运算 符 左 边 的 运算 对 象 向 右 移 动 运算 符 右 侧 指定 的 位 
数 。“ 有 符号 ? 右 移 位 运算 符 使 用 了 “符号 扩展 ”: AEE, MEAN 
AO; 若 值 为 负 ， 则 在 高 位 插入 1。Java 也 添加 了 一 种 “无 符号 ” 右 移 位 运 
GAT (>>>) ， 它 使 用 了 “ 零 扩 展 ”: 无 论 正 负 ， 都 在 高 位 插入 0。 这 一 运 
算 符 是 C 或 C++ 没有 的 。 


藻 对 char，byte 或 者 Short 进行 移 位 处 理 ， 那 么 在 移 位 进行 之 前 ， 它 们 会 

自动 转换 成 一 个 int。 只 有 右 侧 的 5 个 低位 才 会 用 到 。 这 样 可 防止 我 们 在 
一 个 int 数 里 移动 不 切实 际 的 位 数 。 奋 对 一 个 long 值 进行 处 理 ， 最 后 得 到 
的 结果 也 是 long。 此 时 只 会 用 到 右 侧 的 6 个 低位 ， 防 止 移动 超过 long 值 里 
现成 的 位 数 。 但 在 进行 “无 符号 ” 右 移 位 时 ， 也 可 能 遇 到 一 个 问题 。 大 对 
byte 或 short 值 进行 右 移 位 运算 ， 得 到 的 可 能 不 是 正确 的 结果 “(Java 1.0 和 
Java 1.1 特 别 突出 ) 。 它 们 会 自动 转换 成 int 类 型 ， 并 进行 右 移 位 。 但 “ 零 
扩展 ”不 会 发 生 ， 所 以 在 那些 情况 下 会 得 到 -1 的 结果 。 可 用 下 面 这 个 例 

子 检 测 自 己 的 实现 方案 : 


//: URShift.java 











// Test of unsigned right shift 
public class URShift { 
public static void main(String[] args) { 
int i = -1; 
i >>>= 10; 
System.out.println(i); 
long 1 = -1; 
l >>>= 10; 
System.out.println(1); 
short s = -1; 
S >>>= 10; 


System.out.printin(s); 


byte b = -1; 
b >>>= 10; 
System.out.println(b); 


} 
} ///:~ 


移 位 可 与 等 号 (<<= 或 >>= 或 >>>=) 组 合 使 用 。 此 时 ， 运 算 符 左边 的 值 
会 移动 由 右边 的 值 指定 的 位 数 ， 再 将 得 到 的 结果 赋 回 左边 的 值 。 


了 如 何 应 用 涉及 “ 按 位 ”操作 的 所 有 运算 符 ，L 
它们 的 效果 : 


//: BitManipulation.java 





// Using the bitwise operators 

import java.util.*; 

public class BitManipulation { 

public static void main(String[] args) { 

Random rand = new Random(); 
int 1 = rand.nextInt(); 
int j = rand.nextInt(); 
pBinInt("-1", -1); 
pBinInt("+1", +1); 
int maxpos = 2147483647; 
pBinInt("maxpos", maxpos); 


int maxneg = -2147483648; 


pBinInt("maxneg", maxneg); 
pBinInt("i", i); 

pBinInt("~i", ~i); 

pBinInt("-i", -i); 

pBinInt("j", j); 

pBinInt("i & j", i & j); 
pBiniInt("i | j", i | j); 
pBinInt("i ^ j", i ^ j); 
pBinInt("i << 5", i << 5); 
pBinInt("i >> 5", i >> 5); 
pBinInt("(~i) >> 5", (~i) >> 5); 
pBinInt("i >>> 5", i >>> 5); 
pBinInt("(~i) >>> 5", (~i) >>> 5); 
long 1 = rand.nextLong(); 

long m = rand.nextLong(); 
pBinLong("-1L", -1L); 
pBinLong("+1L", +1L); 

long 11 = 9223372036854775807L; 
pBinLong("maxpos", 11); 

long lln = -9223372036854775808L; 
pBinLong("maxneg", lln); 
pBinLong("1", 1); 


pBinLong("~1", ~1); 


pBinLong("-1", -1); 
pBinLong("m", m); 
pBinLong("1l & m", 1 & m); 
pBinLong("l1 | m", 1 | m); 
pBinLong("1 ^ m", 1^ m); 
pBinLong("1 << 5", 1 << 5); 
pBinLong("1 >> 5", 1 >> 5); 
pBinLong("(~1) >> 5", (~1) >> 5); 
pBinLong("1 >>> 5", 1 >>> 5); 
pBinLong("(~1) >>> 5", (~1) >>> 5); 
} 
static void pBinInt(String s, int i) { 

System.out.printin( 

s+", int: "+ i+", binary: "); 
System.out.print(" i) 
for(int j = 31; j >=0; j--) 

if(((1 << j) & i) != 0) 

System.out.print("1"); 
else 
System.out.print("0"); 

System.out.println(); 


} 


static void pBinLong(String s, long 1) 


System.out.printin( 

s + ", long: "+ 1+ ", binary: "); 
System.out.print(" ee 
for(int i = 63; i >=0; i--) 

if(((1L << i) & 1) != 0) 

System.out.print("1"); 
else 
System.out.print("0"); 
System.out.println(); 
} 
} ///:~ 


程序 末尾 调用 了 两 个 方法 : pBinInt() 和 pBinLong()。 它 们 分 别 操作 一 个 
int 和 long 值 ， 并 用 一 种 二 进 制 格式 输出 ， 同 时 附 有 简要 的 说 明文 字 。 目 
前 ， 可 暂时 忽略 它们 具体 的 实现 方案 。 


大 家 要 注意 的 是 System.out.printO 的 使 用 ， 而 不 是 System.out.println0)。 
print() 方 法 不 会 产生 一 个 新 行 ， 以 便 在 同一 行 里 罗列 多 种 信息 。 

除 展示 所 有 按 位 运算 符 针 对 int 和 long 的 效果 之 外 ， 本 例 也 展示 了 int 和 
long 的 最 小 值 、 最 大 值 、+1 和 -1 值 ， 使 大 家 能 体会 它们 的 情况 。 注 意 高 
位 代表 正 负 号 : 0 为 正 ，1 为 负 。 下 面 列 出 int 部 分 的 输出 : 


-1, int: -1, binary: 





11111111111111111111111111111111 
+1, int: 1, binary: 


00000000000000000000000000000001 


maxpos, int: 2147483647, binary: 
01111111111111111111111111111111 
maxneg, int: -2147483648, binary: 
10000000000000000000000000000000 
1, int: 59081716, binary: 
00000011100001011000001111110100 
~i, int: -59081717, binary: 
11111100011110100111110000001011 
-i, int: -59081716, binary: 
11111100011110100111110000001100 
j, int: 198850956, binary: 
00001011110110100011100110001100 
i & j, int: 58720644, binary: 
00000011100000000000000110000100 
i | j, int: 199212028, binary: 
00001011110111111011101111111100 
i ^ j, int: 140491384, binary: 
00001000010111111011101001111000 
1 << 5, int: 1890614912, binary: 
01110000101100000111111010000000 
i >> 5, int: 1846303, binary: 
00000000000111000010110000011111 


(~i) >> 5, int: -1846304, binary: 


11111111111000111101001111100000 
i >>> 5, int: 1846303, binary: 

00000000000111000010110000011111 
(~i) >>> 5, int: 132371424, binary: 


00000111111000111101001111100000 


数字 的 二 进 制 形式 表现 为 “有 符号 2 的 补 值 ”。 
3.1.9 三 元 if-else 运 算 符 


这 种 运算 符 比 较 罕见 ， 因 为 它 有 三 个 运算 对 象 。 但 它 确实 属于 运算 符 的 
一 种 ， 因 为 它 最 终 也 会 生成 一 个 值 。 这 与 本 章 后 一 市 要 讲述 的 普通 if- 
else 语 句 是 不 同 的 。 表 达 式 采取 下 述 形式 : 


布尔 表达 式 ? 值 0: 值 1 


奉 “ 布 尔 表达 式 ” 的 结果 为 tue， 就 计算 “ 值 0”*”， 而 且 它 的 结果 成 为 最 终 由 
运算 符 产 生 的 值 。 但 耕 “ 布 尔 表 达 式 ”的 结果 为 false， 计 算 的 束 是 “ 值 1”， 
而 且 它 的 结果 成 为 最 终 由 运算 符 产 生 的 值 。 


当然 ， 也 可 以 换 用 普通 的 if-else 语 句 〈 在 后 面 介绍 )， 但 三 元 运算 符 更 
加 简洁 。 尺 管 C 引 以 为 傲 的 束 是 它 是 一 种 简练 的 语言 ， 而 且 三 元 运算 符 
的 引入 多 半 就 是 为 了 体现 这 种 高 效率 的 编程 ， 但 假 右 您 打算 频 索 用 它 ， 
还 是 要 先 多 作 一 些 思量 一 一 它 很 容易 就 会 产生 可 读 性 极 差 的 代码 。 


可 将 条 件 运 算 符 用 于 自己 的 “副作用 ”， 或 用 于 它 生 成 的 值 。 但 通常 都 应 
We 因为 那样 做 可 将 运算 符 与 if-else 明 确 区 别 开 。 下 面 便 是 一 
Ul Fs 


























Static int ternary(int i) { 
return i < 10 ? i * 100 :i* 10; 


} 


可 以 看 出 ， 假 设 用 普通 的 让 else 结 构 写 上 述 代 码 ， 代 码 量 会 比 上 面 多 出 
许多 。 如 下 所 示 : 


static int alternative(int i) { 
if (i < 10) 

return i * 100; 
return i * 10; 

} 


但 第 二 种 形式 更 易 理解 ， 而 且 不 要 求 更 多 的 录入 。 所 以 在 挑选 三 元 运算 
符 时 ， 请 务必 权衡 一 下 利 浆 。 





3.1.10 EFRI 


在 C 和 C++ 里 ， 逗 写 不 仅 作 为 函数 昌 变量 列表 的 分 卫 符 使 用 ， 也 作为 进 
行 后 续 计 算 的 一 个 运算 符 使 用 。 在 Java 里 需要 用 到 逗号 的 唯一 场所 就 是 
for 循 环 ， 本 章 稍 后 会 对 此 详 加 解释 。 


3.1.11 字 串 运算 符 + 


这 个 运算 符 在 Java 里 有 一 项 特殊 用 途 : 连接 不 同 的 字 串 。 这 一 点 已 在 前 
面 的 例子 中 展示 过 了 。 尽 管 与 + 的 传统 意义 不 符 ， 但 用 + 来 做 这 件 事 情 仍 
然 是 非常 自然 的 。 在 C++ 里 ， 这 一 功能 看 起 来 非常 不 错 ， 所 以 引入 了 一 
项 “运算 符 过 载 ?机 制 ， 以 便 C++ 程 序 员 为 几乎 所 有 运算 符 增 加 特殊 的 含 
义 。 但 非常 不 幸 ， 与 C++ 的 另外 一 些 限 制 结合 ， 运 算 符 过 载 成 为 一 种 非 
常 复杂 的 特性 ， 程 序 员 在 设计 自己 的 类 时 必须 对 此 有 周到 的 考虑 。 与 

C++ 相 比 ， 尽 管 运算 符 过 载 在 Java 里 更 易 实现 ， 但 迄今 为 止 仍然 认为 这 
0 0 
载运 算 符 。 


我 们 注意 到 运用 “String 。 +” 时 一 些 有 趣 的 现象 。 若 表达 式 以 一 个 String 起 
头 ， 那 么 后 续 所 有 运算 对 象 都 必须 是 字 串 。 如 下 所 示 ; 





























intx =0,y=1,z=2; 


String sString = "x, y, Z "; 
System.out.printIn(sString + x + y + z); 


在 这 里 ，Java 编 译 程序 会 将 x，y 和 z 转 换 成 它们 的 字 串 形式 ， 而 不 是 先 
把 它们 加 到 一 起 。 然 而 ， 如 果 使 用 下 述 语句 : 


System.out.printIn(x + sString); 


ISA SAAS avait Shea A OE RAS EK xe RR — 
$) 。 因 此 ， 如 果 想 通过 “加 号 ?连接 字 串 《使 用 Java 的 早期 版 本 ) ， 请 
e aa a a 
MWR- TFP) 


3.1.12 运算 符 和 常规 操作 规则 

使 用 运算 符 的 一 个 缺点 是 括号 的 运用 经 和 党 容易 搞 错 。 即 使 对 一 个 表达 式 
如 何 计算 有 丝 守 不 确定 的 因素 ， 都 容易 混 消 括号 的 用 法 。 这 个 问题 在 
Java 里 仍然 存在 。 


在 C 和 C++ 中 ， 一 个 特别 常见 的 错误 如 下 : 





while(x = y) { 
Ien 


} 


程序 的 意图 是 测试 是 否 “ 相 等 ”(==) ， 而 不 是 进行 赋值 操作 。 在 C 和 
C++ 中 ， 大 是 一 个 非 零 值 ， 那 么 这 种 赋值 的 结果 肯定 是 true。 这 样 使 可 
能 得 到 一 个 无 限 循 环 。 在 Java 里 ， 这 个 表达 式 的 结果 并 不 是 布尔 值 ， 而 
编译 器 期 望 的 是 一 个 布尔 值 ， 而 且 不 会 从 一 个 int 数 值 中 转换 得 来 。 所 以 
在 编译 时 ， 系 统 整 会 提示 出 现 错误 ， 有 效 地 阻止 我 们 进一步 运行 程序 。 
所 以 这 个 缺点 在 Java 里 永远 不 会 造成 更 严重 的 后 果 。 唯 一 不 会 得 到 编译 
错误 的 时 候 是 x 和 y 都 为 布尔 值 。 在 这 种 情况 下 ，x = y 属 于 合法 表达 式 。 
而 在 上 述 情况 下 ， 则 可 能 是 一 个 错误 。 


在 C 和 C++ 里 ， 类 似 的 一 个 问题 是 使 用 按 位 AND 和 OR， 而 不 是 逻辑 AND 
和 OR。 按 位 AND 和 OR 使 用 两 个 字符 之 一 〈& 或 〉， 而 逻辑 AND 和 OR 











使 用 两 个 相同 的 字符 RRR 。 就 象 <=” 和 “==” 一 样 ， 刍 入 一 个 字符 
当然 要 比 键入 两 个 简单 。 在 Java 里 ， 编 译 器 同样 可 防止 这 一 点 ， 因 为 它 
不 允许 我 们 强行 使 用 一 种 并 不 属于 的 类 型 。 

3.1.13 造型 运算 符 

SER” (Cast) 的 作用 是 “与 一 个 模型 匹配 ?”。 在 适当 的 时 候 ，Java 会 将 
一 种 数据 类 型 自动 转换 成 另 一 种 。 例 如 ， 假 设 我 们 为 浮 点 变量 分 配 一 个 
整数 值 ， 计 算 机 会 将 int 自 动 转换 成 float。 通 过 造型 ， 我 们 可 明确 设置 这 
种 类 型 的 转换 ， 或 者 在 一 般 没 有 可 能 进行 的 时 候 强 迫 它 进行 。 


为 进行 一 次 造型 ， 要 将 括号 中 希望 的 数据 类 型 (包括 所 有 修改 符 〉 置 于 
其 他 任何 值 的 左 侧 。 下 面 是 一 个 例子 : 





void casts() { 

int i = 200; 

long | = (long)i; 
long 12 = (long)200; 
} 


正如 您 看 到 的 那样 ， 既 可 对 一 个 数值 进行 造型 处 理 ， 亦 可 对 一 个 变量 进 
行 造型 处 理 。 但 在 这 儿 展 示 的 两 种 情况 下 ， 造 型 均 是 多 余 的 ， 因 为 编译 
器 在 必要 的 时 候 会 自动 进行 int 值 到 long 值 的 转换 。 当 然 ， 仍 然 可 以 设置 
一 个 造型 ， 提 醒 上 自己 留意 ， 也 使 程序 更 清楚 。 在 其 他 情况 下 ， 造 型 只 有 
在 代码 编译 时 才 显 出 重要 性 。 


在 C 和 C++ 中 ， 造 型 有 时 会 让 人 头痛 。 在 Java 里 ， 造 型 则 是 一 种 比较 安 

全 的 操作 。 但 是 ， 知 进行 一 种 名 为 “缩小 转换 ”(Narrowing Conversion) 
的 操作 (也 就 是 说 ， 脚 本 是 能 容纳 更 多 信息 的 数据 类 型 ， 将 其 转换 成 容 
量 较 小 的 类 型 〉， 此 时 就 可 能 面临 信息 丢失 的 危险 。 此 时 ， 编 译 占 会 强 
迫 我 们 进行 造型 ， 就 好 象 说 :“ 这 可 能 是 一 件 危 险 的 事情 一 一 如 有 果 您 想 

让 我 不 顾 一 切 地 做 ， 那 么 对 不 起 ， 请 明确 造型 。” 而 对 于 “放大 转 

换 ”(Widening conversion) ， 则 不 必 进 行 明确 造型 ， 因 为 新 类 型 肯定 能 
容纳 原来 类 型 的 信息 ， 不 会 造成 任何 信息 的 丢失 。 














Java 人 允许 我 们 将 任何 主 类 型 “造型 ?为 其 他 任何 一 种 主 类 型 ， 但 布尔 值 
(bollean) 要 除外 ， 后 者 根本 不 允许 进行 任何 造型 处 理 。“ 类 ”不 允许 进 
行 造型 。 为 了 将 一 种 类 转换 成 男 一 种 ， 必 须 采 用 特殊 的 方法 ( 字 串 是 一 
种 特殊 的 情况 ， 本 书后 面 会 讲 到 将 对 象 造型 到 一 个 类 型 “家 族 ” 里 ;， 例 
Qn, “橡树 ”可 造型 为 “ 树 ”， 反 之 亦 然 。 但 对 于 其 他 外 来 类 型 ， 如 “ 宕 
石 ”， 则 不 能 造型 为 “ 树 ”) 。 


1. 字面 值 


最 开始 的 时 候 ， 知 在 一 个 程序 里 插入 “字面 值 ”〈Literal) , 9a Ae asia As 
能 准确 知道 要 生成 什么 样 的 类 型 。 但 在 有 些 时 候 ， 对 于 类 型 却 是 暧昧 不 
清 的 。 寿 发 生 这 种 情况 ， 必 须 对 编译 占 加 以 适当 的 “指导 ”。 方 法 是 用 与 
字面 值 天 联 的 字符 形式 加 入 一 些 额外 的 信息 。 下 面 这 段 代 码 向 大 家 展示 
了 这 些 字符 。 




















//: Literals.java 


class Literals { 

char c = Oxffff; // max char hex value 
byte b = Ox7f; // max byte hex value 
short s = Ox7fff; // max short hex value 
int i1 = Ox2f; // Hexadecimal (lowercase) 
int 12 = OX2F; // Hexadecimal (uppercase) 
int 13 = 0177; // Octal (leading zero) 
// Hex and Oct also work with long. 
long n1 = 200L; // long suffix 
long n2 = 2001; // long suffix 
long n3 = 200; 


//! long 16(200); // not allowed 


float fi = 1; 

float f2 = 1F; // float suffix 
float f3 = 1f; // float suffix 
float f4 = 1e-45f; // 10 to the power 
float f5 = 1e+9f; // float suffix 
double d1 = 1d; // double suffix 
double d2 = 1D; // double suffix 
double d3 = 47e47d; // 10 to the power 


LT is 


十 六 进 制 (Base 16) 一 一 它 适 用 于 所 有 整数 数据 类 型 一 一 用 一 个 前 置 的 
0x 或 0X 指 示 。 并 在 后 面 跟随 采用 大 写 或 小 写 形式 的 0-9 以 及 arf。 知 试图 
将 一 个 变量 初始 化 成 超出 自身 能 力 的 一 个 值 〈 无 论 这 个 值 的 数值 形式 如 
何 ) ， 编 译 器 就 会 同 我 们 报告 一 条 出 错 消 上 息 。 注 意 在 上 述 代码 中 ， 最 大 
的 十 六 进 制 值 只 会 在 char， byte 以 及 short 号 上 出 现 。 若 超 出 这 一 限制 ， 

编译 器 会 将 值 自 动 变 成 一 个 int， 并 告诉 我 们 需要 对 这 一 次 赋值 进行 “ 缩 
小 造型 "?。 这 样 一 来 ， 我 们 就 可 清 者 楚 获 知 自己 已 超载 了 边界 。 


八进制 (Base 8) 是 用 数字 中 的 一 个 前 置 0 以 及 0-7 的 数位 指示 的 。 在 C， 
C++ 或 者 Java 中 ， 对 二 进 制 数 字 没 有 相应 的 “字面 ?表示 方法 。 


字面 值 后 的 尾随 字符 标志 着 它 的 类 型 。 奉 为 大 写 或 小 写 的 L， 代 表 
long; 大 写 或 小 写 的 F， 代 表 float; 大 写 或 小 写 的 D， 则 代表 double。 


指数 总 是 采用 一 种 我 们 认为 很 不 直观 的 记号 方法 1.39e-47f。 在 科学 与 
工程 学 领域 ，“e” 代 表 上 自然 对 数 的 基数 ， 约 等 于 2.718 (Java 一 种 更 精确 
的 double 值 采用 Math.E 的 形式 ) 。 它 在 象 <1.39xe 的 -47 次 方 ”* 这 样 的 指数 
表达 式 中 使 用 ， 意 味 着 “1.39x2. 718 的 -47 次 方 ”。 然而 ， 自 FORTRAN 语 
言 发 明 后 ， 人 们 自然 而 然 地 觉得 e 代 表 “10 多 少 次 梭 ”?”。 这 种 做 法 显得 颇 
为 古 怪 ， 因 为 FORTRAN 最 初 面 加 的 是 科学 与 工程 设计 领域 。 TS 
SR, EAU ae EEA BASE CERO) 。 但 不 管 
RE, 这 种 特别 的 表达 方法 在 C，C++ 以 及 现在 的 Java 中 顽固 地 保留 i 




















了 。 所 以 倘若 您 习惯 将 e 作 为 自然 对 数 的 基数 使 用 ， 那 么 在 Java 中 看 到 
象 “1.39e-47f” 这 样 的 表达 式 时 ， 请 转换 您 的 思维 ， 从 程序 设计 的 角度 思 
考 它 ;， 它 真正 的 含义 是 “1.39x10 的 -47 次 方 ”。 


©: John Kirkham 这 样 写 道 “我 最 早 于 1962 年 在 一 部 IBM 1620 机 器 上 
使 用 FORTRAN I。 那 时 一 一 包括 60 年 代 以 及 70 年 代 的 早期 ， 
FORTRAN 一 直 都 是 使 用 大 写字 母 。 之 所 以 会 出 现 这 一 情况 ， 可 能 是 由 
于 早期 的 输入 设备 大 多 是 老式 电 传 打字 机 ， 使 用 5 位 Baudot 码 ， 那 种 码 
并 不 具备 小 写 能 力 。 乘 暴 表 达 式 中 的 玉 ' 也 肯定 是 大 写 的 ， 所 以 不 会 与 
HAM BFE ae RETR, Jag DIRE) SN. SERS RING 
其 实 很 简单 ， 就 是 工 xponential WR, Bl tee ak ea, RITA 
统 的 基数 一 一 一 般 都 是 10。 当 时 ， 八 进 制 也 在 程序 员 中 广泛 使 用 。 尽 管 
我 自己 未 看 到 它 的 使 用 ， 但 假若 我 在 乘 蝴 表 达 式 中 看 到 一 个 八进制 数 
字 ， 就 会 把 它 认 作 Base 8。 我 记得 第 一 次 看 到 用 小 写 ‘@’ 表 示 指 数 是 在 70 
年 代 末 期 。 我 当时 也 觉得 它 极 易 产生 混淆 。 所 以 说 ， 这 个 问题 完全 是 自 
己 “ 洪 入 ?FORTRAN 里 去 的 ， 并 非 一 开始 残 有 。 如 果 你 真 的 想 使 用 自然 
对 数 的 基数 ， 实 际 有 现成 的 函数 可 供 利 用 ， 但 它们 都 是 大 写 的 。” 


注意 如 果 编译 器 能 够 正确 地 识别 类 型 ， 就 不 必 使 用 尾随 字符 。 对 于 下 述 
语句 : 


long n3 = 200; 


它 并 不 存在 含混 不 清 的 地 方 ， 所 以 200 后 面 的 一 个 L 大 可 省 去 。 然 而 ， 对 
于 下 述 语句 : 


float f4 = 1e-47f: //10HY FEA 


编译 器 通常 会 将 指数 作为 双 精 度数 (double〉 处 理 ， 所 以 假如 没有 这 个 
ae 就 会 收 到 一 条 出 错 提示 ， 告 诉 我 们 须 用 一 个 “造型 ”将 double 转 
成 float。 


2. 转型 


大 家 会 发 现 假 各 对 主 数据 类 型 执行 任何 算术 或 按 位 运算 ， 只 要 它们 * 比 
int 小 ”〈 即 char，byte 或 者 short) ， 那 么 在 正式 执行 运算 之 前 ， 那 些 值 会 
自动 转换 成 int。 这 样 一 来 ， 最 终生 成 的 值 就 是 int 类 型 。 所 以 只 要 把 一 个 
值 赋 回 较 小 的 类 型 ， 就 必须 使 用 “造型 ”。 此 外 ， 由 于 是 将 值 赋 回 给 较 小 














的 类 型 ， 所 以 可 能 出 现 信息 丢失 的 情况 ) 。 通 常 ， 表 达 式 中 最 大 的 数据 
类 型 是 决定 了 表达 式 最 终结 果 大 小 的 那个 类 型 。 车 将 一 个 float 值 与 一 个 
double 值 相 乘 ， 结 果 就 是 double; 如 将 一 个 int 和 一 个 long 值 相 加 ， 由 结 
果 为 long。 


3.1.14 Java 没 有 “sizeof” 


在 C 和 C++ 中 ，sizeofO 运 算 符 能 满足 我 们 的 一 项 特殊 需要 : 获知 为 数据 
项 目 分 配 的 字符 数量 。 在 C 和 C++ 中 ，size0 最 常见 的 一 种 应 用 就 是 “ 移 

植 ”。 不 同 的 数据 在 不 同 的 机 器 上 可 能 有 不 同 的 大 小 ， 所 以 在 进行 一 些 
对 大 小 敏感 的 运算 时 ， 程 序 员 必须 对 那些 类 型 有 多 大 做 到 心中 有 数 。 例 
如 ， 一 台 计 算 机 可 用 32 位 来 保存 整数 ， 而 男 一 台 只 用 16 位 保存 。 显 然 ， 

在 第 一 人 台 机 器 中 ， 程 序 可 保存 更 大 的 值 。 正 如 您 可 能 已 经 想到 的 那样 ， 

移植 是 令 C 和 C++ 程序 员 帆 为 头痛 的 一 个 问题 。 


Java 不 需要 sizeof() 运 算 符 来 满足 这 方面 的 需要 ， 因 为 所 有 数据 类 型 在 所 
有 机 器 的 大 小 都 是 相同 的 。 我 们 不 必 考 虑 移植 问题 一 Java 本 喘 就 是 一 
种 “与 平台 无 关 ” 的 语言 。 


3.1.15 复习 计算 顺序 














在 我 举办 的 一 次 增 训 班 中 ， 有 人 抱怨 运算 符 的 优先 顺序 太 难 记 了 。 一 名 
学 生 推 荐 用 一 句 话 来 帮助 记忆 : “Ulcer Addicts Really Like C A lot”, 
即 “ 尝 疡 患者 特别 喜欢 (维生素 〉C”。 
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当然 ， 对 于 移 位 和 按 位 运算 符 ， 上 表 并 不 是 完美 的 助 记 方法 ， 但 对 于 其 
他 运算 来 说 ， 它 确实 很 管用 。 


3.1.16 运算 符 总 结 


下 面 这 个 例子 向 大 家 展示 了 如 何 随同 特定 的 运算 符 使 用 主 数据 类 型 。 从 
根本 上 说 ， 它 是 同一 个 例子 反 反复 复 地 执行 ， 只 是 使 用 了 不 同 的 主 数据 
类 型 。 文 件 编译 四 不 会 报错 ， 因 为 那些 会 导 到 错误 的 行 已 用 儿 变 成 了 注 
RAR. 


//: AllOps.java 





// Tests all the operators on all the 
// primitive data types to show which 
// ones are accepted by the Java compiler. 
class AllOps { 
// To accept the results of a boolean test: 
void f(boolean b) {} 
void boolTest(boolean x, boolean y) { 
// Arithmetic operators: 
//! x=x y; 
//! x=x/ y; 
//! Xx= x % y; 
//! x=x + y; 


TEN X=X- y; 


//! x+; 


//! X--} 
//! x = +y; 
//! x = -y; 


// Relational and logical: 
//! f(x > y); 
//! f(x >= y); 
//! f(x < y); 


//! f(x <= y); 


f(x == y); 
f(x != y); 
F(ly); 

X = x && y; 
x =X || y; 


// Bitwise operators: 
//! x = ~y; 


X= Xx & y; 


//! x= x <<1; 
//! x = x >> 1; 
//! x = x >> 1; 


// Compound assignment: 


//! x += y; 


//! x -= y; 
//! x *= y; 
//! x /= y; 
//! X% y; 


//! x <<= 1; 
//! x >>= 1; 


//! x >>> 1; 


x & y; 
x ^= y; 
x |= y; 
// Casting: 


//! char c = (char)x; 
//! byte B = (byte)x; 
//! short s = (short)x; 
//! int i = (int)x; 

//! long 1 = (long)x; 
//\ float f = (float)x; 


//! double d = (double)x; 


void charTest(char x, char y) { 
// Arithmetic operators: 


x = (char)(x * y); 


x = (char)(x / y); 
x = (char)(x % y); 
x = (char)(x + y); 
x = (char)(x - y); 
X++; 
X--; 


x = (char)+y; 


x 
lI 


(char)-y; 
// Relational and logical: 
f(x > y); 
f(x >= y); 
f(x < y); 
f(x <= y); 
f(x == y); 
f(x != y); 
//! f(!x); 
//! f(x && y); 
//! f(x || y); 
// Bitwise operators: 


x= (char)~y; 


X (char)(x & y); 


x = (char)(x | y); 


x 
lI 


(char)(x ^ y); 


x = (char)(x << 1); 
x = (char)(x >> 1); 
x = (char)(x >>> 1); 


// Compound assignment: 


x += y; 
x -= y; 

x *= y; 
x /= y; 
x %= y; 
x <<= 1; 
x >>= 1; 
x >>>= 1; 
x &= y; 
x ^= y; 

x |= y; 
// Casting: 


//! boolean b = (boolean)x; 
byte B = (byte)x; 

short s = (short)x; 

int i = (int)x; 

long 1 = (long)x; 

float f = (float)x; 


double d = (double)x; 


} 


void byteTest(byte x, 


byte y) { 


// Arithmetic operators: 


X 


X 


X 


x 
lI 


x 
lI 


x 
lI 


x 
lI 


(byte)(x* y); 
(byte)(x / y); 
(byte)(x % y); 


(byte)(x + y); 


1 
< 
— 


(byte) (x 


(byte)+ y; 


(byte)- y; 


// Relational and logical: 


f(x 
f(x 
f(x 
f(x 
f(x 
f(x 
//! 
//! f(x 


//! f(x 


> y); 

>= y); 
< y); 

say); 
== y); 
I= y); 
f(!x); 
&& y); 


II y); 


// Bitwise operators: 


x = (byte)~y; 

x = (byte)(x & y); 

x = (byte)(x | y); 

x = (byte)(x ^ y); 

x = (byte)(x << 1); 
x = (byte) (x >> 1); 
x = (byte)(x >>> 1); 


// Compound assignment: 


x += y; 

x -= yY; 

x *= y; 

x /= y; 

x %= y; 

x <<= 1; 
x >>= 1; 
x >>>= 1; 
x &= y; 

x ^= y; 

x |= y; 
// Casting: 


//! boolean b = (boolean)x; 
char c = (char)x; 


short s = (short)x; 


int i = (int)x; 


long 1 = (long)x; 


float f = (float)x; 


double d = (double)x; 


} 


void shortTest(short x, 


short y) { 


// Arithmetic operators: 


X = 
x = (short) (x 
x = (short) (x 
x = (short) (x 
x = (short) (x 
xt++; 

X--; 

x = (short)+y; 
x = (short)-y; 


/ 


(short)(x * y); 


// Relational and logical: 


f(x > y); 
f(x >= y); 
f(x < y); 
f(x <= y); 
F(x == y); 


F(x != y); 


//! F(!x); 
//\ F(X && y); 
//\ f(x || y); 


// Bitwise operators: 


x = (short)-~y; 


x = (short)(x & y); 

x = (short)(x | y); 

x = (short)(x ^ y); 

x = (short)(x << 1); 
x = (short)(x >> 1); 
x = (short)(x >>> 1); 


// Compound assignment: 


x += y; 

x -= yY; 

x *= y; 

x /= y; 

x %= y; 

x <<= 1; 
x >>= 1; 
x >>>= 1; 
x & y; 

x ^= y; 


x |= y; 


// Casting: 
//! boolean b = (boolean)x; 
char c = (char)x; 
byte B = (byte)x; 
int i = (int)x; 
long 1 = (long)x; 
float f = (float)x; 
double d = (double)x; 
} 
void intTest(int x, int y) { 
// Arithmetic operators: 
x=x* y; 
x=x/y; 


X =X %  y; 


X++; 
X--; 
x = +y; 
Ry; 


// Relational and logical: 


f(x > y); 


f(x >= y); 


f(x <= y); 
f(x == y); 
f(x != y); 
//\ F(1x); 


//\ F(X && y); 
//! f(x II y); 
// Bitwise operators: 


x = ~y; 


X =X >> 1; 
x = x >>> 1; 


// Compound assignment: 


xX += y; 
x -= y; 
x *= y; 
x /= y; 
x %= y; 
x <<= 1; 


x >>= 1; 


x >>>= 1) 


x &= y; 
x ^= y; 
x |= y; 
// Casting: 


//! boolean b = (boolean)x; 
char c = (char)x; 
byte B = (byte)x; 
short s = (short)x; 
long 1 = (long)x; 
float f = (float)x; 
double d = (double)x; 
} 
void longTest(long x, long y) { 
// Arithmetic operators: 
xX =X * y; 
X=x/ yy; 


X=X%Y; 


XY 


// Relational and logical: 


f(x > y); 
f(x >= y); 
f(x < y); 
F(x <= y); 
f(x == y); 
f(x != y); 
//! F(x); 


//! F(X && y); 

//! f(x || y); 

// Bitwise operators: 
x = ~y; 
X =X &Yy; 


x= xX | y; 


X = X >> 1; 
x = x >>> 1; 
// Compound assignment: 


x += y; 


x /= y; 
x %= y; 
x <<= 1; 
x >>= 1; 
x >>>= 1; 
x &= y; 
x ^= y; 
x |= y; 
// Casting: 
//! boolean b = (boolean)x; 
char c = (char)x; 
byte B = (byte)x; 
short s = (short)x; 
int i = (int)x; 
float f = (float)x; 
double d = (double)x; 
} 
void floatTest(float x, float y) { 
// Arithmetic operators: 
x=x* y; 


x=x/y; 


X++; 
X--; 
x = +y; 
x = -y; 


// Relational and logical: 


f(x > y); 
f(x >= y); 
f(x < y); 
f(x <= y); 
人 
f(x != y); 
//! F(x); 


//! f(x && y); 

A/! f(x || y); 

// Bitwise operators: 
//! x = ~y; 

//! x= x & y; 

//! x=x | y; 

//! x= x ry; 

// x= x <<1; 

//! x = x >> 1; 


//! x = x >> 1; 


// Compound assignment: 


x += y; 
Xey, 
x *= y; 
x /= y; 
x %= y; 


//! x <<= 1; 
//! x >> 1; 


//! x >>> 1; 


//! x & y; 
//! x^ y; 
//! x |= y; 
// Casting: 


//! boolean b = (boolean)x; 
char c = (char)x; 
byte B = (byte)x; 
short s = (short)x; 
int i = (int)x; 
long 1 = (long)x; 
double d = (double)x; 
} 
void doubleTest(double x, double y) { 


// Arithmetic operators: 


X++; 
X--; 
x = +y; 
x= y; 


// Relational and logical: 


f(x > y); 
f(x >= y); 
f(x < y); 
f(x <= y); 
f(x == y); 
f(x 1= y); 
//! f(1x); 


//\ f(x && y); 
//\ f(x II y); 
// Bitwise operators: 


//! x 


~y; 
//! xX =x&y; 


//! x=x | y; 


//! x= Ke Bis 
// x= x <<1; 
//! x = x >> 1; 
//! x = x >>> 1; 


// Compound assignment: 


XY 
x y 
x "= yi 
x /= y; 
x %= y; 


//! x <<= 1; 
//! x >> 1; 


//! x >>> 1; 


//! x & y; 
//! x^ y; 
//! x |= y; 
// Casting: 


//! boolean b = (boolean)x; 
char c = (char)x; 

byte B = (byte)x; 

short s = (short)x; 

int i = (int)x; 


long 1 = (long)x; 


float f = (float)x; 


J 


FA ss 


注意 布尔 值 〈(boolean〉 的 能 力 非常 有 限 。 我 们 只 能 为 其 赋予 tue 和 和 false 
值 。 而 且 可 测试 它 为 真 还 是 为 假 ， 但 不 可 为 它们 再 添加 布尔 值 ， 或 进行 
其 他 其 他 任何 类 型 运算 。 


在 char，byte 和 short 中 ， 我 们 可 看 到 算术 运算 符 的 “转型 ”效果 。 对 这 些 

类 型 的 任何 一 个 进行 算术 运算 ， 都 会 获得 一 个 int 结 果 。 必 须 将 其 明 

确 “ 造 型 * 回 原来 的 类 型 (缩小 转换 会 造成 信息 的 丢失 〉 ， 以 便 将 值 赋 回 

那个 类 型 。 但 对 于 int 值 ， 却 不 必 进 行 造型 处 理 ， 因 为 所 有 数据 都 已 经 属 

于 int 类 型 。 然 而 ， 不 要 放松 警惕 ， 认 为 一 切 事 情 都 是 安全 的 。 如 果 对 两 

人 ZR EW elit HH oR AI PIF A ZR 
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//: Overflow. java 











// Surprise! Java lets you overflow. 
public class Overflow { 
public static void main(String[] args) { 
int big = Ox7fffffff; // max int value 
prt("big = " + big); 
int bigger = big * 4; 
prt("bigger = " + bigger); 
} 
static void prt(String s) { 


System.out.println(s); 


} 
于 LA 


输出 结果 如 下 : 
big = 2147483647 
bigger = -4 


而 且 不 会 从 编译 希 那 里 收 到 出 错 提示 ， 运 行 时 也 不 会 出 现 寞 币 反 应 。 爪 
EHE (Java) 确实 是 很 好 的 东西 ， 但 却 没有 “那么 ?好 ! 


对 于 char，byte 或 者 short， 混 合 赋值 并 不 需要 造型 。 即 使 它们 执行 转型 
操作 ， 也 会 获得 与 直接 算术 运算 相同 的 结果 。 而 在 另 一 方面 ， 将 造型 略 
去 可 使 代码 显得 更 加 简练 。 


大 家 可 以 看 到 ， 除 boolean 以 外 ， 任 何 一 种 主 类 型 都 可 通过 造型 变 为 其 他 
主 类 型 。 同 样 地 ， 当 造型 成 一 种 较 小 的 类 型 时 ， 必 须 留 意 “ 缩 小 转换 ”的 
后 果 。 人 否则 会 在 造型 过 程 中 不 知 不 党 地 丢失 信息 。 























3.2 执行 控制 


Java 使 用 了 C 的 全 部 控制 语句 ， 所 以 假期 您 以 前 用 C 或 C++ 编程 ， 其 中 大 
多 数 都 应 是 非常 熟悉 的 。 大 多 数 程序 化 的 编程 语言 都 提供 了 某 种 形式 的 
控制 语句 ， 这 在 语言 则 通常 是 共通 的 。 在 Java 里 ， 涉 及 的 关键 字 包 括 if- 
else、while、do-while、for 以 及 一 个 名 为 switch 的 选择 语句 。 然 而 ，Java 
并 不 文 持 非 常 有 害 的 goto〈 它 仍 是 解决 某 些 特殊 问题 的 权宜 之 计 ) 。 仍 
然 可 以 进行 象 goto 那 样 的 跳 转 ， 但 比 典 型 的 goto 要 局 限 多 了 。 

3.2.1 HANE 

所 有 条 件 语句 都 利用 条 件 表达 式 的 真 或 假 来 决定 执行 流程 。 条 件 表达 式 
的 一 个 例子 是 A==B。 它 用 条 件 运 算 符 “==” 来 判断 A 值 是 否 等 于 B 值 。 该 
表达 式 返 回 true 或 false。 本 章 早 些 时 候 接 触 到 的 所 有 关系 运算 符 都 可 拿 
来 构造 一 个 条 件 语 句 。 注 意 Java 不 允许 我 们 将 一 个 数字 作为 布尔 值 使 

用 ， 即 使 它 在 C 和 C++ 里 是 允许 的 (真是 非 零 ， 而 假 是 零 ) . BRE 
次 布尔 测试 中 使 用 一 个 非 布 尔 值 比如 在 if(a) 里 ， 那 么 首先 必须 用 一 
个 条 件 表达 式 将 其 转换 成 一 个 布尔 值 ， 例 如 if(a!=0)。 

3.2.2 if-else 


if-else 语 句 或 许 是 控制 程序 流程 最 基本 的 形式 。 其 中 的 else 是 可 选 的 ， 所 
以 可 按 下 述 两 种 形式 来 使 用 让 : 


if( 布 尔 表达 式 ) 
语句 
或 者 
if( 布 尔 表达 式 ) 
语句 
else 


语句 























条 件 必 须 产生 一 个 布尔 结果 。 e 号 结尾 的 一 个 简单 语 
句 ， 要 么 是 一 个 复合 语 折 一 组 简单 语句 。 在 本 书 任 
何 地 方 ，》 只 要 提 及 中 看 名 ”这 个 词 就 有 可 能 包括 简单 或 复合 诡 句 。 


作为 让 else 的 一 个 例子 ， 下 面 这 个 test0) 方 法 可 告诉 我 们 猜测 的 一 个 数字 
位 于 目标 数字 之 上 、 之 下 还 是 相等 : 


static int test(int testval) { 








int result = 0; 

if(testval > target) 
result = -1; 

else if(testval < target) 
result = +1; 

else 
result = 0; // match 


return result; 


} 


最 好 将 流程 控制 语句 缩 进 排列 ， 使 读者 能 方便 地 看 出 起 点 与 终点 。 
1. return 


retum 关 键 字 有 两 方面 的 用 途 : 指定 一 个 方法 返回 什么 值 〈 假 设 它 没有 
void 返回 值 ) ， 并 立即 返回 那个 值 。 可 据 此 改写 上 面 的 testO0) 方 法 ， 使 其 


利用 这 些 特 点 : 


static int test2(int testval) { 


if(testval > target) 


return -1; 
if(testval < target) 
return +1; 


return 0; // match 


ANA I _Eelse, HATER Bllreturn Ja (2 AF AE 

3.2.3 反复 

while，do-while 和 for 控 制 着 循环 ， 有 时 将 其 划分 为 “反复 语句 ”。 除 非 用 
于 控制 反复 的 布尔 表达 式 得 到 * 假 2 的 结果 ， 人 否则 语句 会 重复 执行 下 去 。 
while 循 环 的 格式 如 下 : 

while( 布 尔 表 达 式 ) 

语句 


在 循环 刚 开 始 时 ， 会 计算 一 次 “布尔 表达 式 ” 的 值 。 而 对 于 后 来 每 一 次 额 
外 的 循环 ， 都 会 在 开始 前 重新 计算 一 次 。 


下 面 这 个 简单 的 例子 可 产生 随机 数 ， 直 到 符合 特定 的 条 件 为 止 : 


//: WhileTest.java 














// Demonstrates the while loop 
public class WhileTest { 
public static void main(String[] args) { 
double r = 0; 
while(r < 0.99d) { 


r = Math.random(); 


System.out.println(r); 


} 
A f/f 


它 用 到 了 Math 库 里 的 static〈 静 态 ) 方法 random()。 该 方法 的 作用 是 产生 
0 和 1 之 间 (包括 0， 但 不 包括 1) 的 一 个 double 值 。while 的 条 件 表达 式 意 
思 是 说 : “一 直 循 环 下 去 ， 直 到 数字 等 于 或 大 于 0.99”。 由 于 它 的 随机 
性 ， 每 运行 一 次 这 个 程序 ， 都 会 获得 大 小 不 同 的 数字 列表 。 

3.2.4 do-while 

do-while 的 格式 如 下 : 

do 

语句 

while( 布 尔 表 达 式 ) 

while 和 do-while 唯 一 的 区 别 就 是 do-while 肯 定 会 至 少 执行 一 次 ;也 就 是 
说 ， 至 少 会 将 其 中 的 语句 “过 一 亿 ” 即便 表达 式 第 一 次 便 计算 为 
false。 而 在 while 循 环 结构 中 ， 千 条 件 第 一 次 就 为 false， 那 么 其 中 的 语句 
根本 不 会 执行 。 在 实际 应 用 中 ，while 比 do-while 更 常用 一 些 。 

3.2.5 for 


for 循 环 在 第 一 次 反复 之 前 要 进行 初始 化 。 随 后 ， 它 会 进行 条 件 测试 ， 而 
且 在 每 一 次 反复 的 时 候 ， 进 行 某 种 形式 的 “ 步 进 ”(Stepping) 。for 循 环 
的 形式 如 下 : 

for( 初 始 表达 式 ; 布尔 表达 式 ; 步 进 ) 

语句 

无 论 初始 表达 式 ， 布 尔 表 达 式 ， 还 是 步 进 ， 都 可 以 置 空 。 每 次 反复 前 ， 




















都 要 测试 一 下 布尔 表达 式 。 大 获得 的 结果 是 false， 就 会 继续 执行 紧 跟 在 
for 语 句 后 面 的 那 行 代码 。 在 每 次 循环 的 末尾 ， 会 计算 一 次 步 进 。 


for 循 环 通常 用 于 执行 “计数 ”任务 : 


//: ListCharacters.java 





// Demonstrates "for" loop by listing 
// all the ASCII characters. 
public class ListCharacters { 
public static void main(String[] args) { 
for( char c = 0; c < 128; c++) 
if (c != 26 ) // ANSI Clear screen 
System. out.printin( 
"value: " + (int)c + 
" character: " + c); 
} 
由 AAS 








注意 变量 c 是 在 需要 用 到 和 它 的 时 候 定 义 的 一 一 在 for 循 环 的 控制 表达 式 内 
er ee eee 
| 的 表达 式 。 


以 于 象 C 这 样 传统 的 程序 化 语言 ， 要 求 所 有 变量 都 在 一 个 块 的 开头 定 
义 。 所 以 在 编译 器 创建 一 个 块 的 时 候 ， 它 可 以 为 那些 变量 分 配 空间 。 而 
在 Java 和 C++ 中 ， 则 可 在 整个 块 的 范围 内 分 散 变量 声明 ， 在 真正 需要 的 
地 方才 加 以 定义 。 这 样 便 可 形成 更 目 然 的 编码 风格 ， 也 更 易 理 解 。 


可 在 for 语 句 里 定义 多 个 变量 ， 但 它们 必须 具有 同样 的 类 型 : 














for(int i = 0, j = 1; 


i < 10 && j != 11; 
i++, j++) 


/* body of for loop */; 





HF, fong a) A inte X EA m ij RA forth RERE hill Ze 
达 式 里 定义 变量 的 能 力 。 对 于 其 他 任何 条 件 或 循环 语句 ， 都 不 可 采用 这 
种 方法 。 

1. 逗号 运算 符 

早 在 第 1 章 ， 我 们 已 提 到 了 过 号 运算 符 一 注意 不 是 逗号 分 隔 符 ;后 者 
用 于 分 隔 函 数 的 不 同 自 变 量 。Java 里 唯一 用 到 去 号 运算 符 的 地 方 就 是 for 
循环 的 控制 表达 式 。 在 控制 表达 式 的 初始 化 和 步 进 控制 部 分 ， 我 们 可 使 
用 一 系列 由 逗号 分 隔 的 语句 。 而 且 那 些 语 句 均 会 独立 执行 。 前 面 的 例子 
己 运用 了 这 种 能 力 ， 下 面 则 是 男 一 个 例子 : 


//: CommaOperator.java 








public class CommaOperator { 
public static void main(String[] args) { 
for(int i = 1, j = i + 10; i < 5; 
i++, j=i*2){ 


System.out.printin("i= "+ i +" j=" +j); 


EAs 


输出 如 下 : 


i= 1 j= 11 
i= 2 j= 4 
1= 3 j= 6 
1= 4 j= 8 





大 家 可 以 看 到 ， 无 论 在 初始 化 还 是 在 步 进 部 分 ， 语 句 都 是 顺序 执行 的 。 
此 外 ， 尽 管 初 始 化 部 分 可 设置 任意 数量 的 定义 ， 但 都 属于 同一 类 型 。 


3.2.6 中 断 和 继续 

在 任何 循环 语句 的 主体 部 分 ， 亦 可 用 break 和 continue 控 制 循 环 的 流程 。 
其 中 ，break 用 于 强行 退出 循环 ， 不 执行 循环 中 剩余 的 语句 。 而 continue 
则 停止 执行 当前 的 反复 ， 然 后 退回 循环 起 始 和 ， 开 始 新 的 反复 。 

下 面 这 个 程序 向 大 家 展示 了 break 和 continue 在 for 和 while 循 环 中 的 例子 : 


//: BreakAndContinue. java 














// Demonstrates break and continue keywords 
public class BreakAndContinue { 
public static void main(String[] args) { 
for(int i = 0; i < 100; i++) { 
if(i == 74) break; // Out of for loop 
if(i % 9 != 0) continue; // Next iteration 


System.out.println(1); 


int i = 0; 
// An "infinite loop": 
while(true) { 
i++; 
int j = i * 27; 
if(j == 1269) break; // Out of loop 
if(i % 10 != 0) continue; // Top of loop 


System.out.println(1); 


} 
} ///:~ 


在 这 个 for 循 环 中 ，i 的 值 永 远 不 会 到 达 100。 因 为 一 旦 i 到 达 74，break 语 
句 束 会 中 断 循 环 。 通 常 ， 只 有 在 不 知道 中 断 条 件 何 时 满足 时 ， 才 和 需 象 这 
样 使 用 break。 只 要 i 不 能 被 9 整除 ，continue 语 句 会 使 程序 流程 返回 循环 
的 最 开头 执行 〈 所 以 使 i 值 递增 ) 。 如 果 能 够 整除 ， 则 将 值 显示 出 来 。 
第 二 部 分 癌 大 家 揭示 了 一 个 “无 限 循环 ”的 情况 。 然 和 而， 循环 内 部 有 一 个 
break 语 句 ， 可 中 止 循环 。 除 此 以 外 ， 大 家 还 会 看 到 continue 移 回 循环 项 
部 ， 同 时 不 完成 剩余 的 内 容 〈 所 以 只 有 在 ji 值 能 被 9 整除 时 才 打 印 出 

值 ) 。 和 输出 结果 如 下 : 


0 








27 


36 


45 


54 


63 


72 


10 


20 


30 


40 








之 所 以 显示 0， 是 由 于 0%9 等 于 0。 


无 限 循 环 的 第 二 种 形式 是 for(;;)。 编 译 器 将 while(true) 与 for(;;) 看 作 同 一 
回 事 。 所 以 具体 选用 哪个 取决 于 自己 的 编程 习惯 。 


1. AHA “goto” 


goto 天 键 字 很 早 就 在 程序 设计 语言 中 出 现 。 事 实 上 ，goto 是 汇编 语言 的 
程序 控制 结构 的 始祖 :“ 知 条 件 A， 则 跳 到 这 里 ;人 否则 跳 到 那里 ”。 知 阅 
该 由 几乎 所 有 编译 器 生成 的 汇编 代码 ， 就 会 发 现 程序 控制 里 包含 了 许多 
跳 转 。 然 而 ，goto 是 在 源码 的 级 别 跳 转 的 ， 所 以 招致 了 不 好 的 声誉 。 寿 
程序 总 是 从 一 个 地 方 跳 到 另 一 个 地 方 ， 还 有 什么 办 法 能 识别 代码 的 流程 
呢 ? 随 着 Edsger Dijkstra 邯 名 的 “Goto 有 害 ” 论 的 问世 ，goto 便 从 此 失宠 。 


事实 上 ， 真 正 的 问题 并 不 在 于 使 用 goto， 而 在 于 goto 的 小 用。 而 且 在 一 
些 少 见 的 情况 下 ，goto 是 组 织 控制 流程 的 最 佳 手段 。 


尽管 goto 仍 是 Java 的 一 个 保留 字 ， 但 并 未 在 语言 中 得 到 正式 使 用 ，Java 
没有 goto。 然 而 ， 在 break 和 continue 这 两 个 关键 字 的 和 身上， 我 们 仍然 能 
看 出 一 些 goto 的 影子 。 它 并 不 属于 一 次 跳 转 ， 而 是 中 断 循环 语句 的 一 种 
方法 。 之 所 以 把 它们 纳入 goto 问 题 中 一 起 讨论 ， 是 由 于 它们 使 用 了 相同 
的 机 制 : 标签 。 


“标签 ”是 后 面 跟 一 个 冒号 的 标识 待 ， 束 象 下 面 这 样 : 











label1: 


对 Java 来 说 ， 唯 一 用 到 标签 的 地 方 是 在 循环 语句 之 前 。 进 一 步 说 ， 它 实 

际 需要 紧 靠 在 循环 语句 的 前 方 在 标签 和 循环 之 间 置 入 任何 语句 都 是 

不 明智 的 。 而 在 循环 之 前 设置 标签 的 唯一 理由 是 : RIEL PRE 

男 一 个 循环 或 者 一 个 开关 。 这 是 由 于 break 和 continue 关 键 字 通常 只 中 断 

= 但 和 若 随同 标签 使 用 ， 它 们 就 会 中 断 到 存在 标签 的 地 方 。 如 下 
ZN: 


label1: 
外 部 循环 { 
内 部 循环 { 


Hee 








break; //1 

Wises 

continue; //2 

Ian 

continue label1; //3 

ie 

break label1; //4 

} 

} 

在 条 件 1 中 ，break 中 断 内 部 循环 ， 并 在 外 部 循环 结束 。 在 条 件 2 中 ， 
continue 移 回 内 部 循环 的 起 始 处 。 但 在 条 件 3 中 ，continue label1 却 同时 中 


断 内 部 循环 以 及 外 部 循环 ， 并 移 至 label1 处 。 随 后 ， 它 实际 是 继续 循 
环 ， 但 却 从 外 部 循环 开始 。 在 条 件 4 中 ，break label =P TATA A 


环 ， 并 回 到 labell 处 ， 但 并 不 重新 进入 循环 。 也 就 是 说 ， 它 实际 是 完全 
中 止 了 两 个 循环 。 


下 面 是 for 循 环 的 一 个 例子 : 


//: LabeledFor.java 


// Java's "labeled for loop" 
public class LabeledFor { 
public static void main(String[] args) { 
int i = 0; 
outer: // Can't have statements here 
for(; true ;) { // infinite loop 
inner: // Can't have statements here 
for(; i < 10; it+) { 
Wt ea); 
if(i == 2) { 
prt("continue"); 
continue; 
} 
if(i == 3) { 
prt("break"); 
i++; // Otherwise i never 
// gets incremented. 


break; 


} 
if(i == 7) { 
prt("continue outer"); 
i++; // Otherwise i never 
// gets incremented. 


continue outer; 


} 

if(i == 8) { 
prt("break outer"); 
break outer; 

} 


for(int k = 0; k < 5; k++) { 
if(k == 3) { 
prt("continue inner"); 


continue inner; 


} 


// Can't break or continue 


// to labels here 


} 


static void prt(String s) { 


System.out.println(s); 


} 
Ri) ss 


这 里 用 到 了 在 其 他 例子 中 已 经 定义 的 prt0 方 法 。 


注意 break 会 中 断 for 循 环 ， 而 且 在 抵达 for 循 环 的 末尾 之 前 ， 递 增 表达 式 
不 会 执行 。 由 于 break 跳 过 了 递增 表达 式 ， 所 以 递增 会 在 i==3 的 情况 下 直 
接 执 行 。 在 i==7 的 情况 下 ，continue outer 语 句 也 会 到 达 循 环 顶 部 ， 而 且 
也 会 跳 过 递增 ， 所 以 它 也 是 直接 递增 的 。 

下 面 是 输出 结果 : 


1 =0 


continue inner 
i=1 

continue inner 
i ie 

continue 

i=3 

break 

i=4 

continue inner 
i=5 

continue inner 


i=6 


continue Inner 

i=/7 

continue outer 

i= 8 

break outer 
WRIA break ” outer 语句 ， 就 没有 办 法 在 一 个 内 部 循环 里 找到 出 外 部 循 
环 的 路 笃 。 这 是 由 于 break 本 里 只 能 中 断 最 内 层 的 循环 (对 于 continue 同 
样 如 此 ) 。 
当然 ， 知 想 在 中 断 循环 的 同时 退出 方法 ， 简 单 地 用 一 个 retum 即 可 。 


下 面 这 个 例子 回 大 家 展示 了 市 标签 的 break 以 及 continue 语 句 在 while 循 环 
中 的 用 法 : 


//: Labeledwhile.java 





// Java's "labeled while" loop 
public class Labeledwhile { 
public static void main(String[] args) { 
int i = 0; 
outer: 
while(true) { 
prt("Outer while loop"); 
while(true) { 
i++; 


prt("i =" + i); 


if(i == 1) { 
prt("continue"); 
continue; 

} 

if(i == 3) { 
prt("continue outer"); 
continue outer; 

} 

if(i == 5) { 
prt("break"); 
break; 

} 

if(i == 7) { 
prt("break outer"); 


break outer; 


} 
static void prt(String s) { 


System.out.println(s); 


de 


同样 的 规则 亦 适 用 于 while: 
(1) 简单 的 一 个 continue 会 退回 最 内 层 循环 的 开头 《顶部 ) ， 并 继续 执 
AT 





(2) 带 有 标签 的 continue 会 到 达标 签 的 位 置 ， 并 重新 进入 紧 接 在 那个 标签 
后 面 的 循环 。 


(3) break 会 中 断 当 前 循环 ， 并 移 离 当前 标签 的 末尾 。 


(4) 带 标签 的 break 会 中 断 当 前 循环 ， 并 移 离 由 那个 标签 指示 的 循环 的 末 
Eo 


这 个 方法 的 输出 结果 是 一 目 了 然 的 : 


Outer while loop 


i=1 
continue 
i=2 
1= 3 


continue outer 


Outer while loop 


i=4 
i=5 
break 


Outer while loop 


1=6 


i=7 


break outer 





大 家 要 记 住 的 重点 是 : Java EE — m t H Be HT EA aE 
JEA, i EAE F DT BAK aE eS EE al TR o 

fEDijkstral!)“GotoA WWP, ERRIRE MmsFgoto. Baas 
签 在 一 个 程序 里 数量 的 增多 ， 他 发 现 产 生 错 误 的 机 会 也 越 来 越 多 。 标 签 
和 goto 使 我 们 难于 对 程序 作 静 态 分 析 。 这 是 由 于 它们 在 程序 的 执行 流程 
中 引入 了 许多 “怪圈 ”。 但 幸运 的 是 ，Java 标 签 不 会 造成 这 方面 的 问题 ， 
因为 它们 的 活动 场所 已 被 限 死 ， 不 可 通过 特别 的 方式 到 处 传递 程序 的 控 
制 权 。 由 此 也 引出 了 一 个 有 趣 的 问题 : 通过 限制 语句 的 能 力 ， 反 而 能 使 
一 项 语言 特性 更 加 有 用 。 

3.2.7 FPR 


TER” (Switch) 有 时 也 被 划分 为 一 种 “选择 语句 ”*。 根 据 一 个 整数 表达 
式 的 值 ，switch 语 句 可 从 一 系列 代码 选 出 一 段 执 行 。 它 的 格式 如 下 : 


switch( 整 数 选择 因子 ) { 

case 整数 值 1 : 语句 ; break; 
case 整数 值 2 : 语句 ; break; 
case 整数 值 3 : 语句 ; break; 
case 整数 值 4 : 语句 ; break; 
case 整数 值 5 : 语句 ; break; 


/hss 




















default: 语 句 ; 


} 


其 中 , “整数 选择 因子 ?是 一 个 特殊 的 表达 式 ， 能 产生 整数 值 。switch 能 
将 整数 选择 因子 的 结果 与 每 个 整数 值 比较 。 寿 发 现 相符 的 ， 束 执行 对 应 
的 语句 “人 简单 或 复合 语句 ) 。 大 没有 发 现 相符 的 ， 束 执行 default 语 句 。 


在 上 面 的 定义 中 ， 大 家 会 注意 到 每 个 case 均 以 一 个 break 结 尾 。 这 样 可 使 
执行 流程 跳 转 至 switch 主 体 的 末尾 。 这 是 构建 switch 语 句 的 一 种 传统 方 
式 ， 但 break 是 可 选 的 。 若 省 略 break， 会 继续 执行 后 面 的 case 语 句 的 代 
人 码 ， 直 到 过 到 一 个 break 为 止 。 尽 管 通常 不 想 出 现 这 种 情况 ， 但 对 有 经 
验 的 程序 员 来 说 ， 也 许 能 够 善 加 利用 。 注 意 最 后 的 default 语 句 没 有 
break， 因 为 执行 流程 已 到 了 break 的 跳 转 目的 地 。 当 然 ， 如 果 考 虑 到 编 
程 风格 方面 的 原因 ， 完 全 可 以 在 default 语 句 的 末尾 放置 一 个 break， 尺 管 
它 并 没有 任何 实际 的 用 处 。 

Switch 语句 是 实现 多 路 选择 的 一 种 易 行 方式 〈 比 如 从 一 系列 执行 路 径 中 
挑选 一 个 ) 。 但 它 要 求 使 用 一 个 选择 因子 ， 并 且 必 须 是 int 或 char 那 样 的 
整数 值 。 人 例如， 假若 将 一 个 字 串 或 者 浮 点 数 作为 选择 因子 使 用 ， 那 么 它 
| aa 对 于 非 整 数 类 型 ， 则 必须 使 用 一 系列 
if 语 FA) o 

下 面 这 个 例子 可 随机 生成 字母 ， 并 判断 它们 是 元 音 还 是 辅音 字母 : 


//: VowelsAndConsonants. java 




















// Demonstrates the switch statement 
public class VowelsAndConsonants { 
public static void main(String[] args) { 
for(int i = 0; i < 100; i++) { 

char c = (char)(Math.random() * 26 + ‘a'); 
System.out.print(c + ": "); 
Switch(c) { 
case ‘a': 


case ‘e': 


case 'i': 


case 'o': 

case 'u': 
System.out.println("vowel"); 
break; 

case 'y': 

case ‘w': 
System.out.printin( 

"Sometimes a vowel"); 

break; 

default: 
System.out.println("consonant") ; 

} 

} 
} 
} ///:~ 


由 于 Math.random() 会 产生 0 到 1 之 间 的 一 个 值 ， 所 以 只 需 将 其 乘 以 想 获得 
的 最 大 随机 数 〈 对 于 英语 字母 ， 这 个 数字 是 26) ， 青 加 上 一 个 偏 移 量 ， 
得 到 最 小 的 随机 数 。 


尽管 我 们 在 这 儿 表 面 上 要 处 理 的 是 字符 ， 但 switch 语 句 实际 使 用 的 字符 
的 整数 值 。 在 case 语 名 中 ， 用 单 引 写 封闭 起 来 的 字符 也 会 产生 整数 值 ， 
以 便 我 们 进行 比较 。 


请 注意 case 语 句 相 互 间 是 如 何 聚 合 在 一 起 的 ， 它 们 依次 排列 ， 为 一 部 分 
特定 的 代码 提供 了 多 种 匹配 模式 。 也 应 注意 将 break 语 句 置 于 一 个 特定 


否则 控制 流程 会 简单 地 下 移 ， 并 继续 判断 下 一 个 条 件 是 否 
HIF 。 


1. 具体 的 计算 
应 特别 留意 下 面 这 个 语句 : 
char c = (char)(Math.random() * 26 + 'a’); 


Math.random( 会 产生 一 个 double 值 ， 所 以 26 会 转换 成 double 类 型 ， 以 便 
执行 乘法 运算 。 这 个 运算 也 会 产生 一 个 double 值 。 这 意味 着 为 了 执行 加 
法 ， 必 须 无 将 人 a 转 换 成 一 个 double。 利 用 一 个 “造型 >”，double 结 果 会 转换 
回 char。 


我 们 的 第 一 个 问题 是 ， 造 型 会 对 char 作 什么 样 的 处 理 呢 ? 换言之 ， 假 设 
一 个 值 是 29.7， 我 们 把 它 造 型 成 一 个 char， 那 么 结果 值 到 底 是 30 还 是 29 
We? 答案 可 从 下 面 这 个 例子 中 得 到 : 


//: CastingNumbers.java 





// What happens when you cast a float or double 
// to an integral value? 
public class CastingNumbers { 
public static void main(String[] args) { 
double 
above = 0.7, 
below = 0.4; 
System.out.printin("above: " + above); 
System.out.printin("below: " + below); 
System. out.printin( 


"(int )above: " + (int)above); 


System.out .printlLn( 
"(int)below: " + (int)below); 
System. out.printin( 
"(char)('a' + above): " + 
(char)('a' + above)); 
System.out.printin( 
"(char)('a' + below): " + 


(char)('a' + below) ); 


} 
VALLES 
输出 结果 如 下 : 
above: 0.7 
below: 0.4 


(int )above: 0 
(int)below: 0 
(char)('a' + above): a 


(char)('a' + below): a 


所 以 答案 就 是 : 将 一 个 float 或 double 值 造型 成 整数 值 后 ， 总 是 将 小 数 部 
DY ARE”, AEE AAT EAE A EE 


第 二 个 问题 与 Math.randomO0 有 关 。 它 会 产生 0 和 1 之 间 的 值 ， 但 是 否 包括 
值 '1' 呢 ?用 正统 的 数学 语言 表达 ， 它 到 底 是 (0,1)，[0,1]，(0,1]， 还 是 
[0,1) 呢 “《〈 方 括号 表示 “包括 ”， 圆 括号 表示 “不 包括 >) ? 同样 地 ， 一 个 示 
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//: RandomBounds. java 


// Does Math.random() produce 0.0 and 1.0? 
public class RandomBounds { 
static void usage() { 
System.err.println("Usage: \n\t" + 
"RandomBounds lower\n\t" + 
"RandomBounds upper"); 
System.exit(1); 
} 
public static void main(String[] args) { 
if(args.length != 1) usage(); 
if(args[0].equals("lower")) { 
while(Math.random() != 0.0) 
; // Keep trying 
System.out.println("Produced ©.0!"); 
} 
else if(args[0].equals("upper")) { 
while(Math.random() != 1.0) 
; // Keep trying 


System.out.println("Produced 1.0!"); 


else 
usage(); 


} 
} ///:~ 


为 运行 这 个 程序 ， 只 需 在 命令 行 键入 下 述 命 令 即 可 : 
java RandomBounds lower 

al 

java RandomBounds upper 


在 这 两 种 情况 下 ， 我 们 都 必须 人 工 中 断 程 序 ， 所 以 会 发 现 
Math.random) “似乎 ?永远 都 不 会 产生 0.0 或 1.0。 但 这 只 是 一 项 实验 而 
己 。 知 想到 0 和 1 之 间 有 2 的 128 次 方 不 同 的 双 精 度 小 数 ， 所 以 如 果 全 部 产 
生 这 些 数字 ， 人 花费 的 时 间 会 远 远 超 过 一 个 人 的 生命 。 当 然 ， 最 后 的 结果 
是 在 Math.random() 的 输出 中 包括 了 0.0。 或 者 用 数字 语言 表达 ， 输 出 值 
范围 是 [0,1)。 





3.3 上 总结 


本 章 总 结 了 大 多 数 程序 设计 语言 都 具有 的 基本 特性 : 计算 、 运 算 符 优先 
顺序 、 类 型 转换 以 及 选择 和 循环 等 等 。 现 在 ， 我 们 作 好 了 相应 的 准备 ， 
可 继续 癌 面 问 对 象 的 程序 设计 领域 迈进 。 在 下 一 重 里 ， 我 们 将 讨论 对 象 
的 初始 化 与 清除 问题 ， 再 后 面 则 讲述 隐藏 的 基本 实现 方法 。 














3.4 练习 

(1) 写 一 个 程序 ， 打 印 出 1 到 100 间 的 整数 。 

(2) ”修改 练习 (1)， 在 值 为 47 时 用 一 个 break 退 出 程序 。 亦 可 换 成 return 试 
试 。 








(3) 创建 一 个 switch 语 句 ， 为 每 一 种 case 都 显示 一 条 消息 。 并 将 switch 置 
入 一 个 for 循 环 里 ， 令 其 尝试 每 一 种 case。 在 每 个 case 后 面 都 放置 一 个 
break， 并 对 其 进行 测试 。 人 然后， 删除 break， 看 看 会 有 什么 情况 出 现 。 
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第 4 章 初始 化 和 清除 

“ 随 着 计算 机 的 进步 ，: 不 安全 "的 程序 设计 已 成 为 造成 编程 代价 高 晶 的 罪 
魁 祸首 之 一 。” 


“初始 化 ?和 ?清除 "是 这 些 安全 问题 的 其 中 两 个 。 许 多 C 程 序 的 错误 都 是 
由 于 程序 员 怎 记 初 始 化 一 个 变量 造成 的 。 对 于 现成 的 库 ， 辱 用 户 不 知道 
如 何 初始 化 库 的 一 个 组 件 ， 残 往往 会 出 现 这 一 类 的 错误 。 清 除 是 力 一 个 
特殊 的 问题 ， 因 为 用 完 一 个 元 又 后 ， 由 于 不 再 关心 ， 所 以 很 容易 把 它 志 
记 。 这 样 一 来 ， 那 个 元 系 占 用 的 资源 会 一 直 保 留 下 去 ， 极 易 产 生 资 源 

《主要 是 内 存 ) ARM aR 


C++ 为 我 们 引入 了 “构建 器 ”的 概念 。 这 是 一 种 特殊 的 方法 ， 在 一 个 对 象 
创建 之 后 目 动 调用 。Java 也 沿用 了 这 个 概念 ， 但 新 增 了 自己 的 “垃圾 收 

集 器 *”， 能 在 资源 不 再 需要 的 时 候 上 自动 释放 它们 。 本 章 将 讨论 初始 化 和 
清除 的 问题 ， 以 及 Java 如 何 提 供 它 们 的 支持 。 





4.1 用 构建 右上 日 动 初始 化 


对 于 方法 的 创建 ， 可 将 其 想象 成 为 自己 写 的 每 个 类 都 调用 一 次 
initialize0。 这 个 名 字 提 醒 我 们 在 使 用 对 象 之 前 ， 应 首先 进行 这 样 的 调 
用 。 但 不 幸 的 是 ， 这 也 意味 着 用 户 必须 记 住 调用 方法 。 在 Java 中 ， 由 于 
提供 了 名 为 “构建 器 ”的 一 种 特殊 方法 ， 所 以 类 的 设计 者 可 担保 每 个 对 象 
都 会 得 到 正确 的 初始 化 。 若 某 个 类 有 一 个 构建 器 ， 那 么 在 创建 对 象 时 ， 
Java 会 自动 调用 那个 构建 器 一 一 甚至 在 用 户 毫 不 知觉 的 情况 下 。 所 以 说 
这 是 可 以 担保 的 ! 

















接着 的 一 个 问题 是 如 何 命名 这 个 方法 。 存 在 两 方面 的 问题 。 第 一 个 是 我 
们 使 用 的 任何 名 字 都 可 能 与 打算 为 条 个 类 成 员 使 用 的 名 字 冲 突 。 第 二 是 
由 于 编译 器 的 黄 任 是 调用 构建 器 ， 所 以 它 必 须知 道 要 调用 是 哪个 方法 。 
C++ 采取 的 方案 看 来 是 最 简单 的 ， 且 更 有 逻辑 性 ， 所 以 也 在 Java 里 得 到 
了 应 用 : 构建 名 的 名 字 与 类 名 相同 。 这 样 一 来 ， 可 保证 象 这 样 的 一 个 方 
法 会 在 初始 化 期 间 目 动 调用 。 











下 面 是 带 有 构建 器 的 一 个 简单 的 类 (大 执行 这 个 程序 有 问题 ， 请 参考 第 
3 章 的 “赋值 ?小 节 ) 。 


//: SimpleConstructor.java 


// Demonstration of a simple constructor 
package c04; 
class Rock { 

Rock() { // This is the constructor 


System.out.println("Creating Rock"); 


} 


public class SimpleConstructor { 


public static void main(String[] args) { 
for(int i = 0; i < 10; i++) 
new Rock(); 


} 
} ///:~ 


现在 ， 一 旦 创建 一 个 对 象 : 
new Rock(); 
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前 ， 对 象 得 到 正确 的 初始 化 。 

请 注意 所 有 方法 首 字 母 小 写 的 编码 规则 并 不 适用 于 构建 器 。 这 是 由 于 构 
建 嚣 的 名 字 必 须 与 类 名 完全 相同 ! 

和 其 他 任何 方法 一 样 ， 构 建 器 也 能 使 用 自 变 量 ， 以 便 我 们 指定 对 象 的 具 
ret rile tere 
量 。 如 下 所 示 : 


class Rock { 








Rock(int i) { 
System.out.printin( 


"Creating Rock number " + i); 


} 
public class SimpleConstructor { 


public static void main(String[] args) { 


for(int i = 0; i < 10; i++) 


new Rock(i); 


利用 构建 器 的 目 变 量 ， 我 们 可 为 一 个 对 象 的 初始 化 设 定 相 应 的 参数 。 举 
个 例子 来 说 ， 假 设 类 Tree 有 一 个 构建 器 ， 它 用 一 个 整数 自 变量 标记 树 的 
高 度 ， 那 么 就 可 以 象 下 面 这 样 创 建 一 个 Tree 对 和 象 : 


tree t = new Tree(12); / 12 英 尺 高 的 树 


奉 Tree(int) 是 我 们 唯一 的 构建 种 ， 那 么 编译 器 不 会 允许 我 们 以 其 他 任何 
方式 创建 一 个 Tree 对 象 。 


构建 名 有 助 于 消除 大 量 涉及 类 的 问题 ， 并 使 代码 更 易 疝 读 。 例 如 在 前 述 
的 代码 段 中 ， 我 们 并 未 看 到 对 initialize() 方 法 的 明确 调用 一 一 那些 方法 在 
概念 上 独立 于 定义 内 容 。 在 Java 中 ， 定 义 和 初 始 化 属于 统一 的 概念 
两 者 缺 一 不 可 。 


构建 器 属于 一 种 较 特 殊 的 方法 类 型 ， 因 为 它 没 有 返回 值 。 这 与 void 返回 
值 存 在 着 明显 的 区 别 。 对 于 void 返回 值 ， 尽 管 方法 本 身 不 会 目 动 返 回 什 
么 ， 但 仍然 可 以 让 它 返 回 另 一 些 东西 。 构 建 器 则 不 同 ， 它 不 仅 什么 也 不 
会 目 动 返回 ， 而 且 根 本 不 能 有 任何 选择 。 知 存在 一 个 返回 值 ， 而 且 假设 
我 们 可 以 目 行 选择 返回 内 容 ， 那 么 编译 器 多 少 要 知道 如 何 对 那个 返回 值 
作 什么 样 的 处 理 。 











4.2 方法 过 载 


在 任何 程序 设计 语言 中 ， 一 项 重要 的 特性 就 是 名 字 的 运用 。 我 们 创建 一 
个 对 象 时 ， 会 分 配 到 一 个 保存 区 域 的 名 字 。 方 法 名 代表 的 是 一 种 具体 的 
行动 。 通 过 用 名 字 描 述 目 己 的 系统 ， 可 使 自己 的 程序 更 易 人 们 理解 和 修 
改 。 它 非 第 象 写 散 文 一 一 目的 是 与 读者 沟通 。 


我 们 用 名 字 引 用 或 描述 所 有 对 象 与 方法 。 春 名 字 选 得 好 ， 可 使 目 己 及 其 
他 人 更 易 理 解 目 己 的 代码 。 


将 人 类 语言 中 存在 细致 差别 的 概念 “映射 ”到 一 种 程序 设计 语言 中 时 ， 会 
出 现 一 些 特殊 的 问题 。 在 日 常生 活 中 ， 我 们 用 相同 的 词 表达 多 种 不 同 的 
含义 一 一 即 词 的 “过 载 *。 我 们 说 “ 洗 衬 衫 ”、“ 洗 车 ”以 及 “ 洗 狗 ”*。 但 耕 强 
MR PTA, BE TREE: “RISE 衬衫 ”、“ 和 车 洗 车 ”以 及 “ 狗 洗 
狗 ”。 这 是 由 于 听众 根本 不 需要 对 执行 的 行动 作 任何 明确 的 区 分 。 人 类 
的 大 多 数 语言 都 具有 很 强 的 “ 见 余 ? 性 ， 所 以 即使 漏 掉 了 几 个 词 ， 仍 然 可 
以 推断 出 含义 。 我 们 不 需要 独一无二 的 标识 符 一 一 可 从 具体 的 语 境 中 推 


论 出 含义 。 


大 多 数 程序 设计 语言 (特别 是 C，》 要 求 我 们 为 每 个 函数 都 设 定 一 个 独 一 
无 二 的 标识 符 。 所 以 绝对 不 能 用 一 个 名 为 print0 的 函数 来 显示 整数 ， 再 
用 另 一 个 printO 显 示 浮 点 数 一 一 每 个 函数 都 要 求 具 备 唯一 的 名 字 。 


在 Java 里 ， 男 一 项 因素 强迫 方法 名 出 现 过 载 情 况 : 构建 器 。 由 于 构建 器 
的 名 字 由 类 名 决定 ， 所 以 只 能 有 一 个 构建 嚣 名称。 但 假 略 我 们 想 用 多 种 
方式 创建 一 个 对 象 呢 ? 例 如， 假设 我 们 想 创 建 一 个 类 ， 令 其 用 标准 方式 
进行 初始 化 ， 另 外 从 文件 里 读 取 信息 来 初始 化 。 此 时 ， 我 们 需要 两 个 构 
Em, -RATE UGE) ， 另 一 个 将 字 串 作为 目 变 量 
用 于 初始 化 对 象 的 那个 文件 的 名 字 。 由 于 都 是 构建 器 ， 所 以 它们 必须 有 
相同 的 名 字 ， 亦 即 类 名 。 所 以 为 了 让 相同 的 方法 名 伴随 不 同 的 目 变 量 类 
型 使 用 , “方法 过 载 ? 是 非常 关键 的 一 项 措施 。 同 时 ， 尽 管 方法 过 载 是 构 
建 右 必需 的 ， 但 它 亦 可 应 用 于 其 他 任何 方法 ， 且 用 法 非常 方便 。 


人 
法 : 






































//: Overloading.java 


// Demonstration of both constructor 
// and ordinary method overloading. 
import java.util.*; 
class Tree { 
int height; 
Tree() { 
prt("Planting a seedling"); 
height = 0; 
} 
Tree(int i) { 
prt("Creating new Tree that is " 
+ i+ " feet tall"); 
height = i; 
} 
void info() { 
prt("Tree is " + height 
+ " feet tall"); 
} 
void info(String s) { 
prt(s + ": Tree is " 


+ height + " feet tall"); 


static void prt(String s) { 


System.out.println(s); 


} 
public class Overloading { 
public static void main(String[] args) { 
for(int i = 0; i < 5; i++) { 
Tree t = new Tree(1i); 
t.info(); 
t.info("overloaded method"); 
} 
// Overloaded constructor: 
new Tree( ); 
} 


E E 





Tree 既 可 创建 成 一 颗 种 子 ， 不 含 任何 自 变 量 ;， 亦 可 创建 成 生长 在 苗 转 中 

的 植物 。 为 文 持 这 种 创建 ， 共 使 用 了 两 个 构建 器 ， 一 个 没有 目 变 量 〈 我 

ce HRE E aK EER A a”) TERED) ， 另 一 个 采用 现 
Jio 


QO: 在 Sun 公 司 出 版 的 一 些 Java 资 料 中 ， 用 简陋 但 很 说 明 问 题 的 词语 称 
呼 这 类 构建 器 一 一 “无 参数 构建 器 ”(no-arg constructors) 。 但 “默认 构建 
器 ”这 个 称呼 已 使 用 了 许多 年 ， 所 以 我 选择 了 它 。 

我 们 也 有 可 能 希望 通过 多 种 途径 调用 info0 方 法 。 人 例如， 假设 我 们 有 一 
条 额外 的 消息 想 显 示 出 来 ， 就 使 用 String 自 变量 ;而 假设 没有 其 他 话 可 
说 ， 就 不 使 用 。 由 于 为 显然 相同 的 概念 赋予 了 两 个 独立 的 名 字 ， 上 所 以 看 














起 来 可 能 有 些 上 古怪 。 幸 运 的 是 ， 方 法 过 载 允 许 我 们 为 两 者 使 用 相同 的 名 
Fe 


4.2.1 区 分 过 载 方法 


右 方 法 有 同样 的 名 字 ，Java 怎 样 知 道 我 们 指 的 哪 一 个 方法 呢 ? 这 里 有 一 
ae vas 每 个 过 载 的 方法 部 必须 采取 独一无二 的 目 变 量 类 型 列 











石 稍微 思考 儿 秒 钟 ， 就 会 想到 这 样 一 个 问题 : 除根 据 目 变量 的 类 型 ， 程 
序 员 如 何 区 分 两 个 同名 方法 的 差异 呢 ? 


即使 自 变 量 的 顺 友 也 足够 我 们 区 分 两 个 方法 (尽管 我 们 通常 不 愿意 采用 
这 种 方法 ， 因 为 它 会 产生 难以 维护 的 代码 ) : 


//: OverloadingOrder.java 














// Overloading based on the order of 
// the arguments. 
public class OverloadingOrder { 
static void print(String s, int i) { 
System.out.printin( 
"String: " + S + 


Mines CAN 


static void print(int i, String s) { 
System.out.printin( 
"int: "+ 工 十 


, String: " + s); 


} 
public static void main(String[] args) { 
print("String first", 11); 
print(99, "Int first"); 
} 
} ///:~ 








两 个 print0 方 法 有 完全 一 致 的 自 变 量 ， 但 顺序 不 同 ， 可 据 此 区 分 它们 。 
4.2.2 主 类 型 的 过 载 

主 〈 数 据 ) 类 型 能 从 一 个 “ 较 小 ”的 类 型 自动 转变 成 一 个 “ 较 大 ”的 类 型 。 
涉及 过 载 问 题 时 ， 这 会 稍微 造成 一 些 混乱 。 下 面 这 个 例子 揭示 了 将 主 类 
型 传递 给 过 载 的 方法 时 发 生 的 情况 : 


//: PrimitiveOverloading.java 











// Promotion of primitives and overloading 
public class PrimitiveOverloading { 

// boolean can't be automatically converted 
static void prt(String s) { 

System.out.println(s); 

} 

void fi(char x) { prt("fi(char)"); } 

void fi(byte x) { prt("fi(byte)"); } 

void fi(short x) { prt("fi(short)"); } 


void fi(int x) { prt("fi(int)"); } 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


fi(long x) { prt("fa(long)"); } 
fi(float x) { prt("fi(float)"); } 
fi(double x) { prt("fi(double)"); } 
f2(byte x) { prt("f2(byte)"); } 
f2(short x) { prt("f2(short)"); } 
f2(int x) { prt("f2(int)"); } 
f2(long x) { prt("f2(long)"); } 
f2(float x) { prt("f2(float)"); } 
f2(double x) { prt("f2(double)"); } 
f3(short x) { prt("f3(short)"); } 
f3(int x) { prt("f3(int)"); } 
f3(long x) { prt("f3(long)"); } 
f3(float x) { prt("f3(float)"); } 
f3(double x) { prt("f3(double)"); } 
f4(int x) { prt("f4(int)"); } 
f4(long x) { prt("f4(long)"); } 
f4(float x) { prt("f4(float)"); } 
f4(double x) { prt("f4(double)"); } 
f5(long x) { prt("f5(long)"); } 
f5(float x) { prt("f5(float)"); } 
f5(double x) { prt("f5(double)"); } 
f6(float x) { prt("f6(float)"); } 


f6(double x) { prt("f6(double)"); } 


void f7(double x) { prt("f7(double)"); } 
void testConstVal() { 
prt("Testing with 5"); 
f1(5);f2(5);f3(5);f4(5);*5(5);f6(5);F7(5); 
} 


void testChar() { 


char x = 'x'; 
prt("char argument:"); 
f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 

} 

void testByte() { 
byte x = 0; 
prt("byte argument:"); 
f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 

} 

void testShort() { 
short x = 0; 
prt("short argument:"); 
f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 

} 

void testInt() { 
int x = 0; 


prt("int argument:"); 


f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 
} 
void testLong() { 
long x = 0; 
prt("long argument:"); 
f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 
} 
void testFloat() { 
float x = 0; 
prt("float argument:"); 
f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 
} 
void testDouble() { 
double x = 0; 
prt("double argument:"); 
f1(x);f2(x);f3(x);f4(x);f5(x);f6(x);f7(x); 
} 
public static void main(String[] args) { 
PrimitiveOverloading p = 
new PrimitiveOverloading(); 
p.testConstVal(); 
p.testChar(); 


p.testByte(); 


.testShort(); 


Ee) 


.testInt(); 


Ee) 


.testLong(); 


5 


.testFloat(); 


Ke) 


.testDouble(); 


Ee) 


} 
Lif] l= 


i EIR SEY A, hes RS BUSS ES bE. AT 
假 厂 可 以 使 用 一 个 过 载 的 方法 ， 束 能 获取 它 使 用 的 int 值 。 在 其 他 所 有 情 
况 下 ， 知 我 们 的 数据 类 型 “小 于 ?方法 中 使 用 的 目 变 量 ， 就 会 对 那 种 数据 
类 型 进行 “转型 处理 。char 获 得 的 效果 稍 有 些 不 同 ， 这 是 由 于 假期 它 没 
有 发 现 一 个 准确 的 char 匹 配 ， 就 会 转型 为 int。 


若 我 们 的 自 变量 * 大 于 "过载 方 法 期 望 的 自 变量 ， 这 时 又 会 出 现 什么 情况 
呢 ? 对 前 述 程序 的 一 个 修改 揭示 出 了 答案 ; 


//: Demotion. java 

















// Demotion of primitives and overloading 
public class Demotion { 
static void prt(String s) { 
System.out.println(s); 
} 
void fi(char x) { prt("fi(char)"); } 
void fi(byte x) { prt("fi(byte)"); } 


void fi(short x) { prt("fi(short)"); } 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


void 


fi(int x) { prt("fi(int)"); } 
fi(long x) { prt("fa(long)"); } 
fi(float x) { prt("fi(float)"); } 
fi(double x) { prt("fi(double)"); } 
f2(char x) { prt("f2(char)"); } 
f2(byte x) { prt("f2(byte)"); } 
f2(short x) { prt("f2(short)"); } 
f2(int x) { prt("f2(int)"); } 
f2(long x) { prt("f2(long)"); } 
f2(float x) { prt("f2(float)"); } 
f3(char x) { prt("f3(char)"); } 
f3(byte x) { prt("f3(byte)"); } 
f3(short x) { prt("f3(short)"); } 
f3(int x) { prt("f3(int)"); } 
f3(long x) { prt("f3(long)"); } 
f4(char x) { prt("f4(char)"); } 
f4(byte x) { prt("f4(byte)"); } 
f4(short x) { prt("f4(short)"); } 
f4(int x) { prt("f4(int)"); } 
f5(char x) { prt("f5(char)"); } 
f5(byte x) { prt("f5(byte)"); } 
f5(short x) { prt("f5(short)"); } 


f6(char x) { prt("f6(char)"); } 


void f6(byte x) { prt("f6(byte)"); } 
void f7(char x) { prt("f7(char)"); } 
void testDouble() { 
double x = 0; 
prt( "double argument:"); 
f1(x);f2( (float )x);f3((long)x);f4((int)x); 
f5((short)x);f6((byte)x);f7((char)x); 
} 
public static void main(String[] args) { 
Demotion p = new Demotion(); 
p.testDouble(); 
} 


} ///:~ 





在 这 里 ， 方 法 采用 了 容量 更 小 、 范 围 更 罕 的 主 类 型 值 。 知 我 们 的 目 变 量 
范围 比 它 宽 ， 就 必须 用 括号 中 的 类 型 名 将 其 转 为 适当 的 类 型 。 如 果 不 这 
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大 家 可 注意 到 这 是 一 种 “缩小 转换 ”。 也 就 是 说 ， 在 造型 或 转型 过 程 中 可 
能 丢失 一 些 信息 。 这 正 是 编译 器 强迫 我 们 明确 定义 的 原因 一 一 我 们 需 明 
MIRIA 有 要 转型 的 愿望 。 

4.2.3 返回 值 过 载 

我 们 很 易 对 下 面 这 些 问 题 感 到 迷惑 : 为 什么 只 有 类 名 和 方法 目 变 量 列 
出 ? 为 什么 不 根据 返回 值 对 方法 加 以 区 分 ? 比如 对 下 面 这 两 个 方法 来 
说 ， 虽 然 它 们 有 同样 的 名 字 和 上 自 变 量 ， 但 其 实 是 很 容易 区 分 的 : 

void f() {} 


























int fQ {} 

若 编译 器 可 根据 上 下 文 〈 语 境 ) 明确 判断 出 含义 ， 比 如 在 int x=f0F, 
那么 这 样 做 完全 没有 问题 。 然 而 ， 我 们 也 可 能 调用 一 个 方法 ， 同 时 忽略 
返回 值 ， 我 们 通常 把 这 称 为 “为 它 的 副作用 去 调用 一 个 方法 "， 因 为 我 们 
关心 的 不 是 返回 值 ， 而 是 方法 调用 的 其 他 效果 。 所 以 假如 我 们 象 下 面 这 
样 调用 方法 : 

f0; 


Java 怎 样 判断 人 的 具体 调用 方式 呢 ? 而 且 别 人 如 何 识别 并 理解 代码 呢 ? 
由 于 存在 这 一 类 的 问题 ， 所 以 不 能 根据 返回 值 类 型 来 区 分 过 载 的 方法 。 


4.2.4 默认 构建 器 

正如 早先 指出 的 那样 ， 默 认 构建 器 是 没有 自 变量 的 。 它 们 的 作用 是 创建 
一 个 “ 空 对 象 "。 若 创建 一 个 没有 构建 器 的 类 ， 则 编译 程序 会 帮 有 我 们 自动 
创建 一 个 默认 构建 器 。 例 如 : 


//: DefaultConstructor.java 

















class Bird { 
int 1; 
} 
public class DefaultConstructor { 
public static void main(String[] args) { 


Bird nc = new Bird(); // default! 


new Bird(); 

它 的 作用 是 新 建 一 个 对 象 ， 并 调用 默认 构建 器 一 一 即使 尚未 明确 定义 一 
个 象 这 样 的 构建 器 。 奉 没有 它 ， 就 没有 方法 可 以 调用 ， 无 法 构建 我 们 的 
对 象 。 然 而 ， 如 采 已 经 定义 了 一 个 构建 妖 《〈 无 论 是 否 有 目 变 量 ) ， 编 译 
程序 都 不 会 帮 我 们 自动 合成 一 个 : 


class Bush { 




















Bush(int i) {} 

Bush(double d) {} 

} 

现在 ， 假 大 使 用 下 述 代码 : 

new Bush(); 

编译 程序 融会 报告 自己 找 不 到 一 个 相符 的 构建 器 。 融 好 象 我 们 没有 设置 
任何 构建 器 ， 编 译 程序 会 说 : “你 看 来 似乎 需要 一 个 构建 器 ， 所 以 让 我 
们 给 你 制造 一 个 吧 。” 但 假如 我 们 写 了 一 个 构建 费 ， 编 译 程序 就 会 
说 :“ 啊 ， 你 已 写 了 一 个 构建 器， 所 以 我 知道 你 想 干什么 ， 如 果 你 不 放 
置 一 个 默认 的 ， 是 由 于 你 打算 省 略 它 。” 

4.2.5 this 关 键 字 


如 果 有 两 个 同类 型 的 对 象 ， 分 别 叫 作 a 和 b， 那 么 您 也 许 不 知道 如 何 为 这 
两 个 对 象 同 时 调用 一 个 f0 方 法: 


class Banana { void f(int i) { /* ... */ } } 








Banana a = new Banana(), b = new Banana(); 








奉 只 有 一 个 名 叫 f() 的 方法 ， 它 怎样 才能 知道 自己 是 为 a 还 是 为 b 调 用 的 


呢 ? 


为 了 能 用 简便 的 、 面 癌 对 象 的 语法 来 书写 代码 一 一 亦 即 “将 消息 发 给 对 
象 ”， 编 译 咒 为 我 们 完成 了 一 些 疾 后 工作 。 其 中 的 秘密 就 是 第 一 个 目 变 
量 传递 给 方法 fy)， 而 且 那 个 自 变 量 是 准备 操作 的 那个 对 象 的 句柄 。 所 以 
前 述 的 两 个 方法 调用 就 变 成 了 下 面 这 样 的 形式 : 





Banana.f(a,1); 
Banana.f(b,2); 


这 是 内 部 的 表达 形式 ， 我 们 并 不 能 这 样 书写 表达 式 ， 并 试图 让 编译 器 接 
受 它 。 但 是 ， 通 过 它 可 理解 幕后 到 帮 发 生 了 什么 事情 。 


假定 我 们 在 一 个 方法 的 内 部 ， 并 希望 获得 当前 对 象 的 句柄 。 由 于 那个 句 
柄 是 由 编译 器 “秘密 ”传递 的 ， 所 以 没有 标识 符 可 用 。 然 而 ， 针 对 这 一 目 
的 有 个 专用 的 关键 字 : this。this 关 键 字 〈 注 意 只 能 在 方法 内 部 使 用 ) 可 
为 已 调用 了 其 方法 的 那个 对 象 生 成 相应 的 句柄 。 可 象 对 竺 其 他 任何 对 象 
句柄 一 样 对 竺 这 个 句柄 。 但 要 注意 ， 假 徊 准备 从 目 己 某 个 类 的 妃 一 个 方 
法 内 部 调用 一 个 类 方法 ， 就 不 必 使 用 this。 只 需 简 单 地 调用 那个 方法 即 

ae a 前 的 this 句 柄 会 自动 应 用 于 其 他 方法 。 所 以 我 们 能 使 用 下 面 这 样 





class Apricot { 

void pick() { /* ... */ } 

void pit() { pick(Q); /* ... */ } 

} 

在 pit() 内 部 ， 我 们 可 以 说 this.pick()， 但 事实 上 无 此 必要 。 编 译 器 能 帮 我 
们 上 自动 完成 。this 关 键 字 只 能 用 于 那些 特殊 的 类 一 一 需 明 确 使 用 当前 对 


象 的 句柄 。 例 如 ， 假 知 您 希望 将 句柄 返回 给 当前 对 和 象 ， 那 么 它 经 稼 在 
retum 语 句 中 使 用 。 


//: Leaf.java 


// Simple use of the "this" keyword 
public class Leaf { 
private int i = 0; 
Leaf increment() { 
i++; 
return this; 
} 
void print() { 
System.out.println("i = " + i); 
} 
public static void main(String[] args) { 
Leaf x = new Leaf(); 
x.increment().increment().increment().print(); 
} 
} ///:~ 


由 于 incrementO 通 过 this 关 键 字 返回 当前 对 象 的 句柄 ， 所 以 可 以 方便 地 对 
同一 个 对 象 执行 多 项 操作 。 
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个 构建 器 ， 以 避免 写 重复 的 代码 。 可 用 this 关 键 字 做 到 这 一 点 。 


通常 ， 当 我 们 说 this 的 时 候 ， 都 是 指 “ 这 个 对 象 ? 或 者 “当前 对 象 ”。 而 且 
它 本 映 会 产生 当前 对 象 的 一 个 句柄 。 在 一 个 构建 妖 中 ， 大 为 其 赋予 一 个 
目 变 量 列 表 ， 那 么 this 关 键 字 会 具有 不 同 的 含义 ; 它 会 对 与 那个 目 变 量 
列表 相符 的 构建 器 进行 明确 的 调用 。 这 样 一 来 ， 我 们 就 可 通过 一 条 直接 








的 途径 来 调用 其 他 构建 器 。 如 下 所 未 : 


//: Flower.java 


// Calling constructors with "this" 
public class Flower { 
private int petalCount = 0; 
private String s = new String("null"); 
Flower(int petals) { 
petalCount = petals; 
System.out.printin( 
"Constructor w/ int arg only, petalCount= " 
+ petalCount); 
} 
Flower(String ss) { 
System.out .printlLn( 


"Constructor w/ String arg only, s=" + ss); 


Flower(String s, int petals) { 
this(petals); 
//! this(s); // Can't call two! 
this.s = s; // Another use of "this" 


System.out.printin("String & int args"); 


} 

Flower() { 
this("hi", 47); 
System.out.printin( 


"default constructor (no args)"); 


} 
void print() { 
//! this(11); // Not inside non-constructor! 
System.out.printin( 
"petalCount = " + petalCount + " s = "+ s); 

} 

public static void main(String[] args) { 
Flower x = new Flower(); 
X.print(); 


} 
ke 


其 中 ， 构 建 器 Flower(String sint petals) 回 我 们 揭示 出 这 样 一 个 问题 : 尽 
管 可 用 this 调 用 一 个 构建 器 ， 但 不 可 调用 两 个 。 除 此 以 外 ， 构 建 句 调用 
必须 是 我 们 做 的 第 一 件 事情 ， 人 否则 会 收 到 编译 程序 的 报错 信息 。 


这 个 例子 也 向 大 家 展示 了 this 的 为 一 项 用 途 。 由 于 目 变 量 s 的 名 字 以 及 成 
员 数 据 s 的 名 字 是 相同 的 ， 所 以 会 出 现 混 消 。 为 解决 这 个 问题 ， 可 用 
this.s 来 引用 成 员 数 据 。 经 常 都 会 在 Java 代 人 码 里 看 到 这 种 形式 的 应 用 ， 本 
书 的 大 量 地 方 也 采用 了 这 种 做 法 。 


在 printO0 中 ， 我 们 发 现 编 译 器 不 让 我 们 从 除了 一 个 构建 器 之 外 的 其 他 任 


























何方 法 内 部 调用 一 个 构建 右 。 
2. static 的 含义 


理解 了 this 关 键 字 后 ， 我 们 可 更 完整 地 理解 static (静态 ) 方法 的 含义 。 
它 意 味 着 一 个 特定 的 方法 没有 this。 我 们 不 可 从 一 个 static 方 法 内 部 发 出 
对 非 static 方 法 的 调用 (注释 @@) ， 尽 管 反 过 来 说 是 可 以 的 。 而 且 在 没有 
任何 对 象 的 前 提 下 ， 我 们 可 针对 类 本 映 发 出 对 一 个 static 方 法 的 调用 。 事 
实 上 ， 那 正 是 static 方 法 最 基本 的 意义 。 它 就 好 象 我 们 创建 一 个 全 局 函数 
的 等 价 物 〈 在 C 语 言 中 ) 。 除 了 全 局 函数 不 允许 在 Java 中 使 用 以 外 ， 知 
将 一 个 static 方 法 置 入 一 个 类 的 内 部 ， 它 就 可 以 访问 其 他 static 方 法 以 及 
static Z Ex- 


O: 有 可 能 发 出 这 类 调用 的 一 种 情况 是 我 们 将 一 个 对 象 句柄 传 到 static 方 
法 内 部 。 随 后 ， 通 过 句柄 〈 此 时 实际 是 this) ， 我 们 可 调用 非 static 方 
法 ， 并 访问 非 static 字 段 。 但 一 般 地 ， 如 果真 的 想 要 这 样 做 ， 只 要 制作 一 
个 普通 的 、 非 static 方 法 即 可 。 


有 些 人 抱怨 static 方 法 并 不 是 “ 面 癌 对 象 " 的 ， 因 为 它们 具有 全 局 函数 的 某 
些 特点 ;利用 static 方 法 ， 我 们 不 必 同 对 象 发 送 一 条 消息 ， 因 为 不 存在 
this。 这 可 能 是 一 个 清楚 的 上 自 变 量 ， 知 您 发 现 目 己 使 用 了 大 量 静 态 方 

法 ， 就 应 重新 思考 上 自己 的 策略 。 然 而 ，static 的 概念 是 非常 实 用 的 ， 许 多 
时 候 都 需要 用 到 它 。 所 以 至 于 它们 是 否 真 的 “ 面 同 对 象 ?， 应 该 留 给 理论 
oe 事实 上， 即使 Smalltalk 在 自己 的 “类 方法 ”里 也 有 类 似 于 static 
FR PG. 




















4.3 清除 : 收尾 和 垃圾 收集 


程序 员 都 知道 “初始 化 ”的 重要 性 ， 但 通常 忘记 清除 的 重要 性 。 毕 竞 ， 谁 
再 要 来 清除 一 个 int 呢 ? 但 是 对 于 库 来 说 ， 用 完 后 简单 地 “释放 "一 个 对 象 
并 非 总 是 安全 的 。 当 然 ，Java 可 用 垃圾 收集 器 回收 由 不 再 使 用 的 对 象 占 
据 的 内 存 。 现 在 考虑 一 种 非常 特殊 且 不 多 见 的 情况 。 假 定 我 们 的 对 象 分 
配 了 一 个 “特殊 ”内存 区 域 ， 没 有 使 用 new。 垃 圾 收集 器 只 知 着 释放 那些 
由 new 分 配 的 内 存 ， 所 以 不 知道 如 何 释 放 对 象 的 “特殊 "内存 。 为 解决 这 
个 问题 ，Java 提 供 了 一 个 名 为 finalize() 的 方法 ， 可 为 我 们 的 类 定义 它 。 

在 理想 情况 下 ， 它 的 工作 原理 应 该 是 这 样 的 : 一 旦 也 圾 收集 器 准备 好 释 
放 对 象 占用 的 存储 空间 ， 它 首先 调用 finalize()， 而 且 只 有 在 下 一 次 垃圾 
收集 过 程 中 ， 才 会 真正 回收 对 象 的 内 存 。 所 以 如 条 使 用 finalize0， 残 可 
以 在 垃圾 收集 期 间 进 行 一 些 重要 的 清除 或 清扫 工作 。 


但 也 是 一 个 潜在 的 编程 陷阱 ， 因 为 有 些 程序 员 〈 特 别 是 在 C++ 开发 背景 
的 ) 刚 开 始 可 能 会 错误 认为 它 束 是 在 C++ 中 为 “破坏 器 ”(Destructor) 使 
用 的 finalize0) 破坏 (清除 ) 一 个 对 象 的 时 候 ， 肯 定 会 调用 这 个 逊 
数 。 但 在 这 里 有 必要 区 分 一 下 C++ 和 Java 的 区 别 ， 因 为 C++ 的 对 象 肯定 
会 被 清除 〈 排 开 编程 错误 的 因素 ) ， 而 Java 对 象 并 非 肯 定 能 作为 垃圾 
被 收集” 去。 或 者 换 句 话说 : 


垃圾 收集 并 不 等 于 “破坏 ”! 


奉 能 时 刻 牢记 这 一 点 ， 踪 到 陷阱 的 可 能 性 就 会 大 大 减少 。 它 意味 着 在 我 
们 不 再 需要 一 个 对 象 之 前 ， 有 些 行动 是 必须 采取 的 ， 而 且 必 须 由 自己 来 
采取 这 些 行动 。Java 并 未 提供 “破坏 器 ?或 者 类 似 的 概念 ， 所 以 必须 创建 
一 个 原始 的 方法 ， 用 它 来 进行 这 种 清除 。 例 如 ， 假 设 在 对 象 创 建 过 程 

中 ， 它 会 将 目 己 描绘 到 屏幕 上 。 如 果 不 从 屏幕 明确 删除 它 的 图 像 ， 那 么 
它 可 能 永远 都 不 会 被 清除 。 若 在 finalize0 里 置 入 某 种 删除 机 制 ， 那 么 假 
BOX Uk SS Fb, ARE tah Babee ERS. (LARK 
Wer, Rtas thE POR. ATEN TS RE: 


我 们 的 对 象 可 能 不 会 当 作 垃圾 被 收兵 ! 


有 时 可 能 发 现 一 个 对 象 的 存储 空间 永远 都 不 会 释放 ， 因 为 自己 的 程序 永 
远 都 接近 于 用 光 空 间 的 临界 点 。 知 程序 执行 结束 ， 而 且 垃 圾 收集 器 一 直 









































都 没有 释放 我 们 创建 的 任何 对 象 的 存储 空间 ， 则 随 着 程序 的 退出 ， 那 些 
资源 会 返回 给 操作 系统 。 这 是 一 件 好 事情 ， 因 为 垃圾 收集 本 喘 也 要 消耗 
一 些 开 销 。 如 永远 都 不 用 它 ， 那 么 永远 也 不 用 文 出 这 部 分 开销 。 


4.3.1 finalizeO 用 途 何在 


此 时 ， 大 家 可 能 已 相信 了 目 己 应 该 将 finalize0 作 为 一 种 常规 用 途 的 清除 
方法 使 用 。 它 有 什么 好 处 呢 ? 


要 记 住 的 第 三 个 重点 是 : 
垃圾 收集 只 跟 内 存 有 关 ! 


也 就 是 说， 垃圾 收集 器 存在 的 唯一 原因 是 为 了 回收 程序 不 再 使 用 的 内 
存 。 所 以 对 于 与 垃圾 收集 有 关 的 任何 活动 来 说 ， 其 中 最 值得 注意 的 是 
finalize() 方 法 ， 它 们 也 必须 同 内 存 以 及 它 的 回收 有 关 。 


但 这 是 否 意 味 着 假如 对 象 包 含 了 其 他 对 象 ，finalize() 束 应 该 明确 释放 那 
些 对 象 呢 ? 答案 是 否定 的 一 盖世 圾 收集 器 会 负责 释放 所 有 对 象 占据 的 内 
存 ， 无 论 这 些 对 象 是 如 何 创建 的 。 它 将 对 finalize() 的 需求 限制 到 特殊 的 
情况 。 在 这 种 情况 下 ， 我 们 的 对 象 可 采用 与 创建 对 象 时 不 同 的 方法 分 配 
一 些 存储 空间 。 但 大 家 或 许 会 注意 到 ，Java 中 的 所 有 东西 都 是 对 象 ， 所 
以 这 到 底 是 怎么 一 回 事 呢 ? 


之 所 以 要 使 用 finalize()， 看 起 来 似乎 是 由 于 有 了 时 需要 采取 与 Java 的 普通 
方法 不 同 的 一 种 方法 ， 通 过 分 配 内 存 来 做 一 些 具 有 C 风 格 的 事情 。 这 主 
要 可 以 通过 “固有 方法 ”来 进行 ， 它 是 从 Java 里 调用 非 Java 方 法 的 一 种 方 
sh (固有 方法 的 问题 在 附录 人 A 讨论 ) 。C 和 C++ 是 目前 唯一 获得 固有 方法 
支持 的 语言 。 但 由 于 它们 能 调用 通过 其 他 语言 编写 的 子 程序 ， 所 以 能 够 
有 效 地 调用 任何 东西 。 在 非 Java 代 码 内 部 ， 也 许 能 调用 C 的 mallocO 系 列 
函数 ， 用 它 分 配 存 储 空间 。 而 且 除 非 调 用 了 free0， 人 否则 存储 空间 不 会 
得 到 释放 ， 从 而 造成 内 存 “ 漏 洞 ?的 出 现 。 当 然 ，free0 是 一 个 C 和 C++ 天 
数 ， 所 以 我 们 需要 在 finalizeO0 内 部 的 一 个 回 有 方法 中 调用 它 。 


读 完 上 述 文字 后 ， 大 家 或 许 已 弄 清楚 了 自己 不 必 过 多 地 使 用 finalize()。 
这 个 思想 是 正确 的 ; 它 并 不 是 进行 普通 清除 工作 的 理想 场所 。 那 么 ， 普 
通 的 清除 工作 应 在 何 处 进行 呢 ? 















































4.3.2 必须 执行 清除 


为 清除 一 个 对 象 ， 那 个 对 象 的 用 户 必 须 在 希望 进行 清除 的 地 点 调用 一 个 
清除 方法 。 这 上 听 起 来 似乎 很 容易 做 到 ， 但 却 与 C++“ 和 破坏 器 ”的 概念 稍 有 
抵触 。 在 C++ 中 ， 所 有 对 象 都 会 破坏 (清除) 。 或 者 换 句 话说 ， 所 有 对 
象 都 “应 该 ”破坏 。 奉 将 C++ 对 象 创建 成 一 个 本 地 对 象 ， 比 如 在 堆栈 中 创 
建 〈 在 Java 中 是 不 可 能 的 ) ， 那 么 清除 或 破坏 工作 就 会 在 “结束 花 括 
号 ?所 代表 的 、 创 建 这 个 对 象 的 作用 域 的 末尾 进行 。 知 对 象 是 用 new 创 建 
的 《类 似 于 Java) ， 那 么 当 程 序 员 调用 C++ 的 delete 命 令 时 〈Java 没 有 这 
个 命令 ) ， 就 会 调用 相应 的 破坏 器 。 若 程序 员 忘 记 了 ， 那 么 永远 不 会 调 
用 破坏 器 ， 我 们 最 终 得 到 的 将 是 一 个 内 存 “ 漏 洞 ?， 另 外 还 包括 对 象 的 其 
他 部 分 永远 不 会 得 到 清除 。 


相反 ，Java 不 允许 我 们 创建 本 地 【局 部 ) 对 象 一 一 无 论 如 何 都 要 使 用 
new。 但 在 Java 中 ， 没 有 “delete” 命 令 来 释放 对 象 ， 因 为 垃圾 收集 器 会 帮 
助 我 们 自动 释放 存储 空间 。 所 以 如 果 站 在 比较 简化 的 了 立场， 我们 可 以 说 
正 是 由 于 存在 垃圾 收集 机 制 ， 所 以 Java 没 有 破坏 器 。 然 而 ， 随 着 以 后 学 
习 的 深入 ， 束 会 知道 垃圾 收集 器 的 存在 并 不 能 完全 消除 对 破坏 器 的 需 
要 ， 或 者 说 不 能 消除 对 破坏 器 代表 的 那 种 机 制 的 需要 而且 绝 对 不 能 
接 调 用 finalize0)， 所 以 应 尽量 避免 用 它 ) 。 知 希望 执行 除 释 放 存 储 空间 
之 外 的 其 他 某 种 形式 的 清除 工作 ， 仍 然 必须 调用 Java 中 的 一 个 方法 。 它 
等 价 于 C++ 的 破坏 器 ， 只 是 没 后 者 方便 。 


finalizeO) 最 有 用 处 的 地 方 之 一 是 观察 垃圾 收集 的 过 程 。 下 面 这 个 例子 加 
大 家 展示 了 垃圾 收集 所 经 历 的 过 程 ， 并 对 前 面 的 陈述 进行 了 总 结 。 


//: Garbage.java 














// Demonstration of the garbage 
// collector and finalization 
class Chair { 
static boolean gcrun = false; 
static boolean f = false; 


static int created = 0; 


static int finalized = 0; 
int 1; 
Chair() { 
1 = ++created; 
if(created == 47) 
System.out.println("Created 47"); 
} 
protected void finalize() { 
if(!gcrun) { 
gcrun = true; 
System.out.printin( 
"Beginning to finalize after " + 
created + " Chairs have been created"); 
} 
if(i == 47) { 
System.out.printin( 
"Finalizing Chair #47, " + 
"Setting flag to stop Chair creation"); 
f = true; 
} 
finalized++; 
if(finalized >= created) 


System.out.printin( 


"All " + finalized + " finalized"); 


J 


public class Garbage { 
public static void main(String[] args) { 
if (args.length == 0) { 
System.err.println("Usage: \n" + 
"java Garbage before\n or:\n" + 
"java Garbage after"); 
return; 
} 
while(!Chair.f) { 
new Chair(); 
new String("To take up space"); 
} 
System.out.printin( 
"After all Chairs have been created:\n" + 
"total created = " + Chair.created + 
", total finalized = " + Chair.finalized); 
if(args[0].equals("before")) { 
System.out.println("gc():"); 
System.gc(); 


System.out.println("runFinalization():"); 


System.runFinalization( ) ; 


} 
System.out.println("bye!"); 
if(args[0].equals("after")) 

System. runFinalizersOnExit(true) ; 


} 
Lif] 1 


上 面 这 个 程序 创建 了 许多 Chair 对 象 ， 而 且 在 垃圾 收集 器 开始 运行 后 的 
某 些 时 候 ， 程 序 会 停止 创建 Chair。 由 于 垃圾 收集 器 可 能 在 任何 时 间 运 
行 ， 所 以 我 们 不 能 准确 知道 它 在 何 时 启动 。 因 此 ， 程 序 用 一 个 名 为 
gcrun 的 标记 来 指出 垃圾 收集 器 是 否 已 经 开始 运行 。 利 用 第 二 个 标记 f， 
Chair 可 告诉 main() 它 应 停止 对 象 的 生成 。 这 两 个 标记 都 是 在 finalizeO) 内 
部 设置 的 ， 它 调用 于 垃圾 收集 期 间 。 


另 两 个 static 变 量 created 以 及 finalized 分 别 用 于 跟踪 已 创建 的 对 
象 数量 以 及 垃圾 收集 器 已 进行 完 收尾 工作 的 对 象 数 量 。 最 后 ， 每 个 
Chair 都 有 它 自己 的 〈( 非 static〉int i， 所 以 能 跟踪 了 解 它 具体 的 编号 是 多 
少 。 编 号 为 47 的 Chair 进 行 完 收尾 工作 后 ， 标 记 会 设 为 true， 最 终结 
Chair 对 象 的 创建 过 程 。 


所 有 这 些 都 在 main0 的 内 部 进行 一 一 在 下 面 这 个 循环 里 : 


while(!Chair.f) { 




















new Chair(); 
new String("To take up space"); 
} 


大 家 可 能 会 疑惑 这 个 循环 什么 时 候 会 停 下 来 ， 因 为 内 部 没有 任何 改变 
Chair.f 值 的 语句 。 然 而 ，finalize() 进 程 会 改变 这 个 值 ， 直 至 最 终 对 编号 








47 的 对 象 进行 收尾 处 理 。 


每 次 循环 过 程 中 创建 的 String 对 象 只 是 属于 额外 的 垃圾 ， 用 于 吸引 垃圾 
一 旦 垃圾 收集 器 对 可 用 内 存 的 容量 感到 “紧张 不 安 "， 束 会 开 
始 关 注 它 。 


运行 这 个 程序 的 时 候 ， 提 供 了 一 个 命令 行 目 变量 “before” 或 者 “after"。 其 
H, “before” 自 变量 会 调用 System.gc() 方 法 《强制 执行 垃圾 收集 器 ) ， 
同时 还 会 调用 System.runFinalization() 方 法 ， 以 便 进 行 收尾 工作 。 这 些 方 
法 都 可 在 Java 1.0 中 使 用 ， 但 通过 使 用 “after”* 自 变量 而 调用 的 
runFinalizersOnExit() 方 法 却 只 有 Java 1.1 及 后 续 版 本 提供 了 对 它 的 支持 

(注释 @)) 。 注 意 可 在 程序 执行 的 任何 时 候 调 用 这 个 方法 ， 而 且 收 尾 程 
序 的 执行 与 垃圾 收集 器 是 否 运 行 是 无 关 的 。 


©: AAW, Java 1.0 采 用 的 垃圾 收集 器 方案 永远 不 能 正确 地 调用 
finalize()。 因 此 ，finalize0 方 法 (特别 是 那些 用 于 关闭 文件 的 ) 事实 上 

经 常 都 不 会 得 到 调用 。 现 在 有 些 文章 声称 所 有 收尾 模块 都 会 在 程序 退出 
的 时 候 得 到 调用 即使 到 程序 中 止 的 时 候 ， 垃 圾 收集 器 仍 未 针对 那些 
对 象 采取 行动 。 这 并 不 是 真实 的 情况 ， 所 以 我 们 根本 不 能 指望 finalize() 
能 为 所 有 对 象 而 调用 。 特 别 地 ，finalizeO 在 Java 1.0 里 几乎 毫 无 用 处 。 

前 面 的 程序 向 我 们 揭示 出 : 在 Java 1.1 中 ， 收 尾 模 块 肯 定 会 运行 这 一 许 
诡 已 成 为 现实 一 但 前 提 是 我 们 明确 地 强制 它 采 取 这 一 操作 。 知 使 用 一 
个 不 是 “before” 或 “after” 的 自 变 量 〈( 如 “none”) ， 那 么 两 个 收尾 工作 都 不 
会 进行 ， 而 且 我 们 会 得 到 象 下 面 这 样 的 输出 : 


Created 47 





























Created 47 


Beginning to finalize after 8694 Chairs have been created 
Finalizing Chair #47, Setting flag to stop Chair creation 
After all Chairs have been created: 

total created = 9834, total finalized = 108 


bye! 


因此 ， 到 程序 结束 的 时 候 ， 并 非 所 有 收尾 模块 都 会 得 到 调用 (注释 

D) 。 为 强制 进行 收尾 工作 ， 可 先 调 用 System.gc()， 再 调用 
System.runFinalization()。 这 样 可 清除 到 目前 为 止 没 有 使 用 的 所 有 对 象 。 
这 样 做 一 个 稍 显 奇 怪 的 地 方 是 在 调用 runFinalization() 之 前 调用 gc()， 这 
看 起 来 似乎 与 Sun 公 司 的 文档 说 明 有 些 抵触 ， 它 宣称 首先 运行 收尾 模 
块 ， 再 释放 存储 空间 。 然 而 ， 知 在 这 里 首先 调用 runFinalization0)， 再 调 
用 gcO0， 收 尾 模块 根本 不 会 执行 。 


D: KERARI, Ava VM 可 能 已 开始 表现 出 不 同 
9 行为 。 


针对 所 有 对 象 ，Java ”1.1 有 时 之 所 以 会 默认 为 跳 过 收尾 工作 ， 是 由 于 它 
认为 这 样 做 的 开销 太 大 。 不 管用 哪 种 方法 强制 进行 垃圾 收集 ， 痢 可 能 注 
意 到 比 没 有 额外 收尾 工作 时 较 长 的 时 间 延 迟 。 





4.4 成 员 初 始 化 


Java 尽 自己 的 全 力 保 证 所 有 变量 都 能 在 使 用 前 得 到 正确 的 初始 化 。 寿 被 
定义 成 相对 于 一 个 方法 的 “局 部 ?变量 ， 这 一 保证 就 通过 编译 期 的 出 错 提 
示 表 现 出 来 。 因 此 ， 如 果 使 用 下 述 代 码 : 


void f() { 
int 1; 
i++; 

} 


PLUS — AM EAN IAS, UPR EARS. SR, SE as 
they KATARE, (EE FER BR EI RK, WEEN ER 
VEU Mi RBE”. er HG Ree he PR PS EL, WIEME HE A e 
/ te 2 h Be Ap ERRER”. 


然而 ， 知 将 基本 类 型 〈 主 类 型 ) 设 为 一 个 类 的 数据 成 员 ， 情 况 就 会 变 得 
稍微 有 些 不 同 。 由 于 任何 方法 都 可 以 初始 化 或 使 用 那个 数据 ， 所 以 在 正 
式 使 用 数据 前 ， 硅 还 是 强 巡 程序 员 将 其 初始 化 成 一 个 适当 的 值 ， 束 可 能 
不 是 一 种 实际 的 做 法 。 然 而 ， 硝 为 其 赋予 一 个 垃圾 值 ， 同 样 是 非常 不 安 
全 的 。 因 此 ， 一 个 类 的 所 有 基本 类 型 数据 成 员 都 会 保证 获得 一 个 初始 
值 。 可 用 下 面 这 段 小 程序 看 到 这 些 值 : 


//: InitialValues.java 














// Shows default initial values 
class Measurement { 

boolean t; 

char c; 


byte b; 


short s; 

int 1; 

long 1; 

float f; 
double d; 

void print() { 


System.out.printin( 


"Data type Inital value\n" + 
"boolean "4 t+ "\n" + 
"char "+Cc+INn + 
"byte We be Ni 
"short "+s + "An" + 
"int "+ i+ "\nNn"+ 
"long Woe A 
"float "4 Ft "An" + 
"double " + d); 
} 
} 


public class InitialValues { 
public static void main(String[] args) { 
Measurement d = new Measurement(); 
d.print(); 


/* In this case you could also say: 


new Measurement().print(); 
se | 
} 
} ///:~ 


输入 结果 如 下 : 


Data type Inital value 


boolean false 


char 

byte 0 
short 0 
int 0 
long 0 
float 0.0 
double 0.0 


其 中 ，Char 值 为 空 (NULL) ， 没 有 数据 打印 出 来 。 


稍 后 大 家 就 会 看 到 : 在 一 个 类 的 内 部 定义 一 个 对 象 句柄 时 ， 如 果 不 将 其 
初始 化 成 新 对 象 ， 那 个 句柄 就 会 获得 一 个 空 值 。 


4.4.1 规定 初始 化 
如 果 想 自己 为 变量 赋予 一 个 初始 值 ， 又 会 发 生 什 么 情况 呢 ? 为 达到 这 个 
目的 ， 一 个 最 直接 的 做 法 是 在 类 内 部 定义 变量 的 同时 也 为 其 赋值 〈 注 意 


在 C++ 里 不 能 这 样 做 ， 尽 管 C++ 的 新 手 们 总 “ 想 ” 这 样 做 ) 。 在 下 面 ， 
Measurement 类 内 部 的 字段 定义 已 发 后 了 变化 ， 提 供 了 初始 值 : 














class Measurement { 


boolean b = true; 
char c = 'x!'; 
byte B = 47; 

short s = Oxff; 

int i = 999; 

long 1 = 1; 

float f = 3.14f; 
double d = 3.14159; 


The, ENS 





亦 可 用 相同 的 方法 初始 化 非 基本 《〈 主 ) 类 型 的 对 象 。 知 Depth 是 一 个 
类 ， 那 么 可 象 下 面 这 样 插入 一 个 变量 并 进行 初始 化 : 


class Measurement { 

Depth o = new DepthQ; 

boolean b = true; 

aan 

4 le AR A ota xe Sa, [AN eA, SPB — 

条 运行 期 错误 提示 ， 告 诉 你 产生 了 名 为 “违例 ”(Exception〉 的 一 个 错误 
(在 第 9 半 详 述 ) 。 

甚至 可 通过 调用 一 个 方法 来 提供 初始 值 : 

class CInit { 


int i = f0; 


flask 
} 


当然 ， 这 个 方法 亦 可 使 用 自 变 量 ， 但 那些 自 变 量 不 可 是 尚未 初始 化 的 其 
他 类 成 员 。 因 此 ， 下 面 这 样 做 是 合法 的 : 








class CInit { 
int i = f0; 
int j = gi); 
Ian 

} 

但 下 面 这 样 做 是 非法 的 : 
class CInit { 
int j = g(i); 
int i = f0; 
ie 

} 


这 正 是 编译 絮 对 “ 辐 前 引用 ”感到 不 适应 的 一 个 地 方 ， 因 为 它 与 初始 化 的 
顺序 有 关 ， 而 不 是 与 程序 的 编译 方式 有 大。 


这 种 初始 化 方法 非常 简单 和 直观 。 它 的 一 个 限制 是 类 型 Measurement 的 
每 个 对 象 都 会 获得 相同 的 初始 化 值 。 有 时 ， 这 正 是 我 们 希望 的 结果 ， 但 
有 时 却 需要 盼望 更 大 的 灵活 性 。 

4.4.2 构建 器 初始 化 


可 考虑 用 构建 器 执行 初始 化 进程 。 这 样 便 可 在 编程 时 获得 更 大 的 灵活 程 








度 ， 因 为 我 们 可 以 在 运行 期 调用 方法 和 采取 行动 ， 从 而 “现场 ”决定 初始 
化 值 。 但 要 注意 这 样 一 件 事 情 : 不 可 妨碍 目 动 初 始 化 的 进行 ， 它 在 构建 
器 进入 之 前 就 会 发 生 。 因 此 ， 假 如 使 用 下 述 代 码 : 

class Counter { 

int i; 

Counter() { i = 7; } 

ee 


那么 首先 会 初始 化 成 零 ， 然 后 变 成 7。 对 于 所 有 基本 类 型 以 及 对 象 句 
柄 ， 这 种 情况 都 是 成 立 的 ， 其 中 包括 在 定义 时 已 进行 了 明确 初始 化 的 那 
些 一 些 。 考 虑 到 这 个 原因 ， 编 译 占 不 会 试 着 强 过 我 们 在 构建 各 任何 特定 
的 场所 对 元 素 进 行 初始 化 ， 或 者 在 它们 使 用 之 前 一 一 初始 化 早已 得 到 了 
RUE GERO) 
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之 前 进行 初始 化 ， 而 且 它 对 于 对 象 来 说 是 强制 进行 的 。 参 见 《Thinking 


in C++) 。 

1. 初始 化 顺序 

在 一 个 类 里 ， 初 始 化 的 顺序 是 由 变量 在 类 内 的 定义 顺序 决定 的 。 即 使 变 
量 定 义 大 量 过 布 于 方法 定义 的 中 间 ， 那 些 变量 仍 会 在 调用 任何 方法 之 前 
得 到 初始 化 一 一 其 至 在 构建 器 调用 之 前 。 例 如 : 


//: OrderOfInitialization.java 




















// Demonstrates initialization order. 

// When the constructor is called, to create a 
// Tag object, you'll see a message: 

class Tag { 


Tag(int marker) { 


System.out.println("Tag(" + marker + ")"); 


J 


class Card { 
Tag t1 = new Tag(1); // Before constructor 
Card() { 
// Indicate we're in the constructor: 
System.out.printin("Card()"); 
t3 = new Tag(33); // Re-initialize t3 
} 
Tag t2 = new Tag(2); // After constructor 
void f() { 
System.out.printin("f()"); 
} 
Tag t3 = new Tag(3); // At end 
} 
public class orderofInitialization { 
public static void main(String[] args) { 
Card t = new Card(); 


t.f(); // Shows that construction is done 


I 


在 Card 中 ，Tag 对 象 的 定义 故意 到 处 散布 ， 以 证 明 它 们 全 部 会 在 构建 器 
进入 或 者 发 生 其 他 任何 事情 之 前 得 到 初始 化 。 除 此 之 外 ，t3 在 构建 器 内 
部 得 到 了 重新 初始 化 。 它 的 输入 结果 如 下 : 


Tag(1) 


Tag(2) 
Tag(3) 
Card() 
Tag(33) 


f() 


因此 ，t3 句 柄 会 被 初始 化 两 次 ， 一 次 在 构建 器 调用 前 ， 一 次 在 调用 期 间 
(第 一 个 对 象 会 被 丢弃 ， 所 以 它 后 来 可 被 当 作 垃 圾 收 掉 ) 。 从 表面 看 ， 
这 样 做 似乎 效率 低下 ， 但 它 能 保证 正确 的 初始 化 一 — 若 定义 了 一 个 过 载 
的 构建 器 ， 它 没有 初始 化 3; 同时 在 t3 的 定义 里 并 没有 规定 “默认 ”的 初 
始 化 方式 ， 那 么 会 产生 什么 后 果 呢 ? 


2. 静态 数据 的 初始 化 


奉 数据 是 静态 的 (static) ， 那 么 同样 的 事情 束 会 友 生 ;如果 它 属于 一 个 
基本 类 型 〈 主 类 型 ) ， 而 且 未 对 其 初始 化 ， 就 会 目 动 获得 自己 的 标准 基 
本 类 型 初始 值 ， 如果 它 是 指向 一 个 对 象 的 句柄 ， 那 么 除非 新 建 一 个 对 

象 ， 并 将 句柄 同 它 连 接 起 来 ， 人 否则 就 会 得 到 一 个 空 值 (NULL) 。 

如 果 想 在 定义 的 同时 进行 初始 化 ， 采 取 的 方法 与 非 静态 值 表面 看 起 来 是 
相同 的 。 但 由 于 static 值 只 有 一 个 存储 区 域 ， 所 以 无 论 创建 多 少 个 对 象 ， 


都 必然 会 遇 到 何 时 对 那个 存储 区 域 进 行 初 始 化 的 问题 。 下 面 这 个 例子 可 
将 这 个 问题 说 更 清楚 一 些 : 


//: StaticInitialization.java 














// Specifying initial values ina 


// class definition. 
class Bowl { 
Bowl(int marker) { 
System.out.println("Bowl(" + marker + ")"); 
} 
void f(int marker) { 


System.out.println("f(" + marker + ")"); 


} 


class Table { 

static Bowl b1 = new Bowl(1); 

Table() { 
System.out.println("Table()"); 
b2.f(1); 

} 

void f2(int marker) { 
System.out.println("f2(" + marker + ")"); 


j; 


static Bowl b2 = new Bowl(2); 
} 
class Cupboard { 

Bowl b3 = new Bowl(3); 


static Bowl b4 = new Bowl(4); 


Cupboard() { 
System.out.println("Cupboard()"); 
b4.f(2); 

} 

void f3(int marker) { 
System.out.println("f3(" + marker + ")"); 


t 


static Bowl b5 = new Bowl(5); 
} 
public class StaticInitialization { 
public static void main(String[] args) { 
System.out.printin( 
"Creating new Cupboard() in main"); 
new Cupboard(); 
System.out.printin( 
"Creating new Cupboard() in main"); 
new Cupboard(); 
t2.f2(1); 
t3.f3(1); 
} 
static Table t2 = new Table(); 
static Cupboard t3 = new Cupboard(); 


} ///:~ 


Bowl 人 允许 我 们 检查 一 个 类 的 创建 过 程 ， 而 Table 和 Cupboard 能 创建 散布 
于 类 定义 中 的 Bowl 的 static 成 员 。 注 意 在 static 定 义 之 前 ，Cupboard 先 创 
建 了 一 个 非 static 的 Bowlb3。 它 的 输出 结果 如 下 : 


Bowl(1) 


Bowl(2) 

Table() 

f(1) 

Bowl (4) 

Bowl(5) 

Bowl(3) 

Cupboard() 

(2) 

Creating new Cupboard() in main 
Bowl(3) 

Cupboard() 

f(2) 

Creating new Cupboard() in main 
Bowl(3) 

Cupboard() 

f(2) 

f2(1) 


f3(1) 


static 初 始 化 只 有 在 必要 的 时 候 才 会 进行 。 如 果 不 创建 一 个 Table 对 象 ， 

而 且 永 远 都 不 引用 Table.b1 或 Table.b2， 那 么 static Bowl b1 和 b2 永 远 都 不 
会 创建 。 然 而 ， 只 有 在 创建 了 第 一 个 Table 对 象 之 后 (或 者 发 生 了 第 一 
次 static 访 问 ) ， 它 们 才 会 创建 。 在 那 以 后 ，static 对 象 不 会 重新 初始 
化 。 


初始 化 的 顺序 是 首先 static《〈 如 果 它 们 尚未 由 前 一 次 对 象 创 建 过 程 初始 
化 )， 接 着 是 非 static 对 象 。 大 家 可 从 输出 结果 中 找到 相应 的 证 据 。 


在 这 里 有 必要 总 结 一 下 对 象 的 创建 过 程 。 请 考虑 一 个 名 为 Dog 的 类 : 

(1) 类 型 为 Dog 的 一 个 对 象 首次 创建 时 ， 或 者 Dog 类 的 static 方 法 / static 字 

和 
Zi) 

(2) 找到 Dog.class 后 〈 它 会 创建 一 个 Class 对 象 ， 这 将 在 后 面 学 到 ) . E 

的 所 有 static 初 始 化 模块 都 会 运行 。 因 此 ，static 初 始 化 仅 发 生 一 次 一 一 

在 Class 对 象 首 次 载 入 的 时 候 。 


(3) 创建 一 个 new Dog0 时 ，Dog 对 象 的 构建 进程 首先 会 在 内 存 堆 
(Heap) 里 为 一 个 Dog 对 象 分 配 足 够 多 的 存储 空间 。 


(4) 这 种 存储 空间 会 清 为 零 ， 将 Dog 中 的 所 有 基本 类 型 设 为 它们 的 默认 值 
《零用 于 数字 ， 以 及 boolean 和 char 的 等 价 设 定 ) 。 


(5) 进行 字段 定义 时 发 生 的 所 有 初始 化 都 会 执行 。 


(6) 执行 构建 器 。 正 如 第 6 章 将 要 讲 到 的 那样 ， 这 实际 可 能 要 求 进行 相当 
多 的 操作 ， 特 别 是 在 涉及 继承 的 时 候 。 


3. 明确 进行 的 静态 初始 化 


Java 允 许 我 们 将 其 他 static 初 始 化 工作 划分 到 类 内 一 个 特殊 的 “static 构 建 
从 句 ”( 有 时 也 叫 作 “静态 块 "”) 里 。 它 看 起 来 象 下 面 这 个 样子 : 


class Spoon { 














static int i; 


static { 
i = 47; 

} 

// ， 


尽管 看 起 来 象 个 方法 ， 但 它 实 际 只 是 一 个 static 关 键 字 ， 后 面 跟随 一 个 方 
法 主体 。 与 其 他 static 初 始 化 一 样 ， 这 段 代码 仅 执行 一 次 一 一 首次 生成 那 
个 类 的 一 个 对 象 时 ， 或 者 首次 访问 属于 那个 类 的 一 个 static 成 员 时 《即便 
从 未 生成 过 那个 类 的 对 象 ) 。 例 如 : 


//: ExplicitStatic.java 





// Explicit static initialization 
// with the "static" clause. 
class Cup { 
Cup(int marker) { 
System.out.printin("Cup(" + marker + ")"); 
} 
void f(int marker) { 


System.out.printin("f(" + marker + ")"); 


} 


class Cups { 
static Cup c1; 


static Cup c2; 


static { 
c1 = new Cup(1); 
c2 = new Cup(2); 
} 
Cups() { 


System.out.println("Cups()"); 


} 
public class ExplicitStatic { 
public static void main(String[] args) { 
System.out.println("Inside main()"); 
Cups.c1.f(99); // (1) 
} 
static Cups x = new Cups(); // (2) 
static Cups y = new Cups(); // (2) 


L/S 


在 标记 为 1) 的 行内 访问 static 对 象 cl 的 时 候 ， 或 在 行 (1) 标 记 为 注释 ， 同 
时 (2) 行 不 标记 成 注释 的 时 候 ， 用 于 Cups 的 static 初 始 化 模块 就 会 运行 。 
若 (0 和 (2) 痢 被 标记 成 注释 ， 则 用 于 Cups 的 satc 初 始 化 进程 永远 不 会 发 


4. 非 静态 实例 的 初始 化 


针对 每 个 对 象 的 非 静态 变量 的 初始 化 ，Java 1.1 提供 了 一 种 类 似 的 语法 
格式 。 下 面 是 一 个 例子 : 


//: Mugs.java 


// Java 1.1 "Instance Initialization" 
Class Mug { 
Mug(int marker) { 
System.out.println("Mug(" + marker + ")"); 
} 
void f(int marker) { 


System.out ,println("f(" + marker + ")"); 


} 


public class Mugs { 
Mug ci; 
Mug c2; 
{ 


c1 = new Mug(1); 


c2 


new Mug(2); 

System.out.println("c1 & c2 initialized"); 
iF 
Mugs() { 

System.out.println("Mugs()"); 


} 


public static void main(String[] args) { 


System.out.println("Inside main()"); 
Mugs X = new Mugs(); 
} 
} ///:~ 


大 家 可 看 到 实例 初始 化 从 句 : 
{ 


c1 = new Mug(1); 
c2 = new Mug(2); 


System.out.printin("c1 & c2 initialized"); 





它 看 起 来 与 静态 初始 化 从 名 极其 相似 ， 只 是 static 关 键 字 从 里 面 消 失 了 。 
为 支持 对 “匿名 内 部 类 ”的 初始 化 《参见 第 7 章 ) ， 必 须 采 用 这 一 语法 格 
式 。 


4.5 数组 初始 化 


在 C 中 初始 化 数组 极 易 出 错 ， 而 且 相 当 厂 烦 。C++ 通 过 “和 集合 初始 化 ”使 
其 更 安全 QRO) 。Java 则 没有 象 C++ 那样 的 “集合 ”概念 ， 因 为 Java 
中 的 所 有 东西 都 是 对 象 。 但 它 确 实 有 目 己 的 数组 ， 通 过 数组 初始 化 来 提 


供 文 持 。 


数组 代表 一 系列 对 象 或 者 基本 数据 类 型 ， 所 有 相同 的 类 型 都 封装 到 一 起 
采用 一 个 统一 的 标识 符 名 称 。 数 组 的 定义 和 使 用 是 通过 方 括号 索引 
运算 符 进 行 的 〈[])。 为 定义 一 个 数组 ， 只 需 在 类 型 名 后 简单 地 跟随 一 
对 空 方 括号 即 可 : 








int[] al; 
也 可 以 将 方 括号 置 于 标识 符 后 面 ， 获 得 完全 一 致 的 结果 : 
int all]; 


这 种 格式 与 C 和 C++ 程序 员 习 惯 的 格式 是 一 致 的 。 然 而 ， 最 “通顺 ”的 也 
许 还 是 前 一 种 语法 ， 因 为 它 指出 类 型 是 “一 个 int 数 组 *”。 本 书 将 沿用 那 种 
格式 。 

编译 器 不 允许 我 们 告诉 它 一 个 数组 有 多 大 。 这 样 便 使 我 们 回 到 了 “名 
柄 ”的 问题 上 。 此 时 ， 我 们 拥有 的 一 切 就 是 指 辐 数 组 的 一 个 句柄 ， 而 且 
尚未 给 数组 分 配 任 何 空间 。 为 了 给 数组 创建 相应 的 存储 空间 ， 必 须 编写 
一 个 初始 化 表达 式 。 对 于 数组 ， 初 始 化 工作 可 在 代码 的 任何 地 方 出 现 ， 
但 也 可 以 使 用 一 种 特殊 的 初始 化 表达 式 ， 它 必须 在 数组 创建 的 地 方 出 
现 。 这 种 特殊 的 初始 化 是 一 系列 由 花 括 号 封闭 起 来 的 值 。 存 储 空 间 的 分 
配 〈 等 价 于 使 用 new) 将 由 编 详 器 在 这 种 情况 下 进行 。 例 如 : 


int[] al = { 1,2, 3,4, 5 }; 
那么 为 什么 还 要 定义 一 个 没有 数组 的 数组 句柄 呢 ? 
int[] a2; 


事实 上 在 Java 中 ， 可 将 一 个 数组 分 配给 另 一 个 ， 所 以 能 使 用 下 述 语句， 








a2 =al; 
我 们 真正 准备 做 的 是 复制 一 个 句柄 ， 就 象 下 面 演 示 的 那样 : 


//: Arrays.java 


// Arrays of primitives. 
public class Arrays { 
public static void main(String[] args) { 
int[] al = { 1, 2, 3, 4, 5 }; 
int[] a2; 
a2 = al; 
for(int i = 0; i < a2.length; i++) 
a2[i]++; 
for(int i = 0; i < ait.length; i++) 
prt( al[ + i+ "] =" + ailil); 
} 
static void prt(String s) { 
System.out.println(s); 
} 
下 


大 家 看 到 al 获得 了 一 个 初始 值 ， 而 a2 没 有 ; a2 将 在 以 后 赋值 一 一 这 种 情 
况 下 是 赋 给 力 一 个 数组 。 


这 里 也 出 现 了 一 些 新 东西 : 所 有 数组 都 有 一 个 本 质 成 员 《〈 无 论 它 们 是 对 
象 数组 还 是 基本 类 型 数组 ) ， 可 对 其 进行 查询 一 一 但 不 是 改变 ， 从 而 获 











知 数 组 内 包含 了 多 少 个 元 素 。 这 个 成 员 束 是 length。 与 C 和 C++ 类 似 ， 由 
于 Java 数 组 从 元 系 0 开 始 计数 ， 所 以 能 索引 的 最 大 元 系 编 号 是 "length- 
1”。 如 超出 边界 ，C 和 C++ 会 “默默 ”地 接受 ， 并 允许 我 们 胡乱 使 用 自己 
的 内 存 ， 这 正 是 许多 程序 错误 的 根源 。 然 而 ，Java 可 保留 我 们 这 受 这 一 
问题 的 损害 ， 方 法 是 一 旦 超过 边界 ， 就 生成 一 个 运行 期 错误 〈 即 一 
个 “违例 ” 这 是 第 9 间 有 的 主题 ) 。 当 然 ， 由 于 需要 检查 每 个 数组 的 访 
问 ， 所 以 会 消耗 一 定 的 时 间 和 多 余 的 代码 量 ， 而 且 没 有 办 法 把 它 关 闭 。 
这 意味 着 数组 访问 可 能 成 为 程序 效率 低下 的 重要 原因 一 一 如 来 它们 在 关 
键 的 场合 进行 。 但 考虑 到 因特网 访问 的 安全 ， 以 及 程序 员 的 编程 效率 ， 
Java 设 计 人 员 还 是 应 该 把 它 看 作 是 值得 的 。 


程序 编写 期 间 ， 如 有 果 不 知 着 在 自己 的 数组 里 需要 多 少 元 素 ， 那 么 又 该 怎 
么 办 呢 ? 此 时 ， 只 需 简单 地 用 new 在 数组 里 创建 元 隶 。 在 这 里 ， 即 使 准 
备 创建 的 是 一 个 基本 数据 类 型 的 数组 ，new 也 能 正常 地 工作 (new 不 会 
创建 非 数 组 的 基本 类 型 ) : 


//: ArrayNew. java 























// Creating arrays with new. 
import java.util.*; 
public class ArrayNew { 
static Random rand = new Random(); 
static int pRand(int mod) { 
return Math.abs(rand.nextInt()) % mod + 1; 
} 
public static void main(String[] args) { 
int[] a; 
a = new int[pRand(20)]; 
prt("length of a = " + a.length); 


for(int i = 0; i < a.length; i++) 


prt( al +i+"]=" + a[i]); 
} 
static void prt(String s) { 
System.out.println(s); 
} 
} ///:~ 


由 于 数组 的 大 小 是 随机 决定 的 (使 用 早先 定义 的 pRand0 方 法 ) ， 所 以 
非常 明显 ， 数 组 的 创建 实际 是 在 运行 期 间 进 行 的 。 除 此 以 外 ， 从 这 个 程 
序 的 输出 中 ， 大 家 可 看 到 基本 数据 类 型 的 数组 元 素 会 自动 初始 化 
成 “ 空 ” 值 (对 于 数值 ， 空 值 就 是 零 ， 对 于 char， 它 是 null;， 而 对 于 


boolean， 它 却 是 false) 。 


当然 ， 数 组 可 能 已 在 相同 的 语句 中 定义 和 初始 化 了 ， 如 下 所 示 : 











int[] a = new int[pRand(20)]; 


FRERE PARSE ARRAS A BL, ABA GVO GH ey AEH new. 
EKE, RNAS VS A) eel, A AERA BE Ze — Th 
组 。 请 大 家 观察 封装 器 类 型 Integer， 它 是 一 个 类 ， 而 非 基 本 数据 类 型 : 


//: ArrayClassObj.java 











// Creating an array of non-primitive objects. 
import java.util.*; 
public class ArrayClassObj { 

static Random rand = new Random(); 

static int pRand(int mod) { 


return Math.abs(rand.nextInt()) % mod + 1; 


} 
public static void main(String[] args) { 
Integer[] a = new Integer[pRand(20)]; 
prt("length of a = " + a.length); 
for(int i = 0; i < a.length; i++) { 
a[i] = new Integer(pRand(500)); 


prt("a[" + 1 + "| 二 " + a[il]); 


} 
static void prt(String s) { 
System.out.println(s); 
} 
} ///:~ 


在 这 儿 ， 甚 至 在 new 调 用 后 才 开 始 创建 数组 : 
Integer[] a = new Integer[pRand(20)]; 


它 只 是 一 个 句柄 数组 ， 而 且 除 非 通过 创建 一 个 新 的 Integer 对 象 ， 从 而 初 
始 化 了 对 象 句 柄 ， 人 否则 初始 化 进程 不 会 结束 : 


ali] = new Integer(pRand(500)); 


但 若 忘记 创建 对 象 ， 就 会 在 运行 期 试图 读 取 空 数组 位 置 时 获得 一 个 < 韦 
例 " 错 误 。 


下 面 让 我 们 看 看 打印 语句 中 String 对 象 的 构成 情况 。 大 家 可 看 到 指向 
A 从 而 产生 一 个 String， 它 代表 着 位 于 对 
部 的 值 。 





亦 可 用 花 括 号 封闭 列表 来 初始 化 对 象 数组 。 可 采用 两 种 形式 ， 第 一 种 是 
Java 1.0 人 允许 的 唯一 形式 。 第 二 种 (等 价 ) 形式 自 Java 1.1 才 开始 提供 文 


F: 


//: ArrayInit.java 


// Array initialization 
public class ArrayInit { 
public static void main(String[] args) { 
Integer[] a = { 
new Integer(1), 
new Integer(2), 
new Integer(3), 
}; 
// Java 1.1 only: 
Integer[] b = new Integer[] { 
new Integer(1), 
new Integer(2), 
new Integer(3), 
}; 
} 
IO 


这 种 做 法 大 多 数 时 候 都 很 有 用 ， 但 限制 也 是 最 大 的 ， 因 为 数组 的 大 小 是 
在 编译 期 间 决 定 的 。 初 始 化 列表 的 最 后 一 个 去 号 是 可 选 的 〈 这 一 特性 使 
长 列表 的 维护 变 得 更 加 容易 ) 。 








数组 初始 化 的 第 二 种 形式 〈Java ”1.1 开始 支持 ) 提供 了 一 种 更 简便 的 语 
法 ， 可 创建 和 调用 方法 ， 获 得 与 C 的 “变量 参数 列表 ”(C 通 种 把 它 简称 
AREER?) 一 致 的 效果 。 这 些 效果 包括 未 知 的 参数 CARE) 数量 以 
及 未 知 的 类 型 〈 如 果 这 样 选择 的 话 ) 。 由 于 所 有 类 最 终 都 是 从 通用 的 根 
类 Object 中 继承 的 ， 所 以 能 创建 一 个 方法 ， 令 其 获取 一 个 Object 数组 ， 
并 象 下 面 这 样 调用 它 : 


//: VarArgs.java 








// Using the Java 1.1 array syntax to create 
// variable argument lists 
class A { int i; } 
public class VarArgs { 
static void f(Object[] x) { 
for(int i = 0; i < x.length; i++) 
System.out.println(x[i]); 
} 
public static void main(String[] args) { 
f(new Object[] { 
new Integer(47), new VarArgs(), 
new Float(3.14), new Double(11.11) }); 
f(new Object[] {"one", "two", "three" }); 


f(new Object[] {new A(), new A(), new A()}); 


} ///:~ 


此 时 ， 我 们 对 这 些 未 知 的 对 象 并 不 能 采取 太 多 的 操作 ， 而 且 这 个 程序 利 
用 目 动 String 转 换 对 每 个 Object 做 一 些 有 用 的 事情 。 在 第 11 章 〈 运 行 期 类 
型 标识 或 RTTI) ， 大 家 还 会 学 习 如 何 调 查 这 类 对 象 的 准确 类 型 ， 使 自 
己 能 对 它们 做 一 些 有 趣 的 事情 。 

4.5.1 多 维 数组 

在 Java 里 可 以 方便 地 创建 多 维 数组 : 


//: MultiDimArray.java 








// Creating multidimensional arrays. 
import java.util.*; 
public class MultiDimArray { 
static Random rand = new Random(); 
static int pRand(int mod) { 
return Math.abs(rand.nextInt()) % mod + 1; 
} 
public static void main(String[] args) { 
int[][] al = { 
{1 Bye Oy hy 
{ 4, 5, 6, }, 
}; 
for(int i = 0; i < ai.length; i++) 
for(int j = 0; j < ai[il].length; j++) 
prt("a1[" + i + [+ 了 + 


"] = " + al[i][j]); 


// 3-D array with fixed length: 
int[][][] a2 = new int[2][2][4]; 
for(int i = 0; i < a2.length; i++) 
for(int j = 0; j < a2[i].length; j++) 
for(int k = 0; k < a2[i][j].length; 
k++) 
prt("a2[" + i + "J[" + 
j + "J]J[" + k + 
"] =" + a2[i][j][k]); 
// 3-D array with varied-length vectors: 
int[][][] a3 = new int[pRand(7)][][]; 
for(int i = 0; i < a3.length; i++) { 
a3[i] = new int[pRand(5)][]; 
for(int j = 0; j < a3[i].length; j++) 
a3[i][j] = new int[pRand(5)]; 
} 
for(int i = 0; i < a3.length; i++) 
for(int j = 0; j < a8[i].length; j++) 
for(int k = 0; k < a3[i][j].length; 
k++) 
prt("a3[" + i + "J[" + 
j + "J][" + k+ 


"] = " + a3[i][j][k]); 


// Array of non-primitive objects: 
Integer[][] a4 = { 
{ new Integer(1), new Integer(2)}, 
{ new Integer(3), new Integer(4)}, 
{ new Integer(5), new Integer(6)}, 
}; 
for(int i = 0; i < a4.length; i++) 
for(int j = 0; j < a4[i].length; j++) 
prt("a4[" + i + "][" + j + 
"] =" + a4[i][j]); 
Integer[][] a5; 
a5 = new Integer[3][]; 
for(int i = 0; i < a5.length; i++) { 
a5[i] = new Integer[3]; 
for(int j = 0; j < a5[i].length; j++) 
a5[i][j] = new Integer(i*j); 
} 
for(int i = 0; i < a5.length; i++) 
for(int j = 0; j < a5[i].length; j++) 
prt("a5[" +i + "][" +j + 
"] =" + aS[i][j]); 
} 


static void prt(String s) { 


System.out.println(s); 


I 
} ///:~ 


用 于 打印 的 代码 里 使 用 了 length， 所 以 它 不 必 依 赖 固定 的 数组 大 小 。 


第 一 个 例子 展示 了 基本 数据 类 型 的 一 个 多 维 数组 。 我 们 可 用 花 括 写 定 出 
数组 内 每 个 天 量 的 边界 : 


int[][] al = { 

{ 1, 2, 3, }, 

{ 4, 5, 6, }, 

} 

每 个 方 括号 对 都 将 我 们 移 至 数组 的 下 一 级 。 


第 二 个 例子 展示 了 用 new 分 配 的 一 个 三 维 数组 。 在 这 里 ， 整 个 数组 都 是 
立即 分 配 的 : 


int[][][] a2 = new int[2][2][4]; 
但 第 三 个 例子 却 回 大 家 揭示 出 构成 矩阵 的 每 个 矢量 都 可 以 有 任意 的 长 


Le 





int[][][] a3 = new int[pRand(7)][][]; 


for(int i = 0; i < a3.length; i++) { 
a3[1] = new int[pRand(5)][]; 
for(int j = 0; j < a3[i].length; j++) 


a3[1][j] = new int[pRand(5)]; 


对 于 第 一 个 new 创 建 的 数组 ， 它 的 第 一 个 元 素 的 长 度 是 随机 的 ， 其 他 元 
素 的 长 度 则 没有 定义 。for 循 环 内 的 第 二 个 new 则 会 填写 元 素 ， 但 保持 第 
三 个 索引 的 未 定 状态 一 一 直到 磁 到 第 三 个 new。 

根据 输出 结果 ， 大 家 可 以 看 到 : 假 徊 没有 明确 指定 初始 化 值 ， 数 组 值 束 
会 目 动 初始 化 成 零 。 

可 用 类 似 的 表 式 处 理 非 基本 类 型 对 象 的 数组 。 这 从 第 四 个 例子 可 以 看 
出 ， 它 癌 我 们 演示 了 用 人 花 括 写 收集 多 个 new 表达 式 的 能 


Integer[][] a4 = { 














{ new Integer(1), new Integer(2)}, 
{ new Integer(3), new Integer(4)}, 
{ new Integer(5), new Integer(6)}, 


}3 





第 五 个 例子 展示 了 如 何 逐 渐 构 建 非 基 本 类 型 的 对 象 数组 : 


Integer[][] a5; 


a5 = new Integer[3][]; 

for(int i = 0; i < a5.length; i++) { 
a5[i] = new Integer[3]; 
for(int j = 0; j < a5[i].length; j++) 


a5[i][j] = new Integer(i*j); 


ij 只 是 在 Integer 里 置 了 一 个 有 趣 的 值 。 


4.6 总 结 


作为 初始 化 的 一 种 具体 操作 形式 ， 构 建 器 应 使 大 家 明确 感受 到 在 语言 中 
进行 初始 化 的 重要 性 。 与 C++ 的 程序 设计 一 样 ， 判 断 一 个 程序 效率 如 

何 ， 关 键 是 看 是 否 由 于 变量 的 初始 化 不 正确 而 造成 了 严重 的 编程 错误 

( 刁 虫 ) 。 这 些 形式 的 错误 很 难 发 现 ， 而 且 类 似 的 问题 也 适用 于 不 正确 
的 清除 或 收尾 工作 。 由 于 构建 器 使 我 们 能 保证 正确 的 初始 化 和 清除 〈 知 
没有 正确 的 构建 器 调用 ， 编 译 器 不 允许 对 象 创 建 ) ， 所 以 能 获得 完全 的 
控制 权 和 安全 性 。 


在 C++ 中 ， 与 “构建 > 相反 的 “破坏 ”(Destruction) 工作 也 是 相当 重要 
的 ， 因 为 用 new 创 建 的 对 象 必 须 明确 地 清除 。 在 Java 中 ， 垃 圾 收集 占 会 
自动 为 所 有 对 象 释 放 内 存 ， 所 以 Java 中 等 价 的 清除 方法 并 不 是 经 常 都 需 
要 用 到 的 。 如 果 不 需 要 类 似 于 构建 器 的 行为 ，Java 的 垃圾 收集 右 可 以 极 
大 简化 编程 工作 ， 而 且 在 内 存 的 管理 过 程 中 增加 更 大 的 安全 性 。 有 些 垃 
圾 收集 器 甚至 能 清除 其 他 资源 ， 比 如 图 形 和 文件 句柄 等 。 然 而 ， 志 圾 收 
集 右 确实 也 增加 了 运行 期 的 开销 。 但 这 种 开销 到 抵 造 成 了 多 大 的 影响 却 
是 很 难看 出 的 ， 因 为 到 目前 为 止 ，Java 解 释 器 的 总 体 运行 速度 仍然 是 比 
较 慢 的 。 随 着 这 一 情况 的 改观 ， 我 们 应 该 能 判断 出 垃圾 收集 器 的 开销 是 
否 使 Java 不 适合 做 一 些 特定 的 工作 (其 中 一 个 问题 是 垃圾 收集 右 不 可 预 
测 的 性 质 ) 。 


由 于 所 有 对 象 都 肯定 能 获得 正确 的 构建 ， 所 以 同 这 儿 讲 述 的 情况 相 比 ， 
构建 费 实 际 做 的 事情 还 要 多 得 多 。 特 别 地 ， 当 我 们 通过 “创作 ”或 “ 继 
承 ” 生 成 新 类 的 时 候 ， 对 构建 的 保证 仍然 有 效 ， 而 且 需 要 一 些 附加 的 语 
法 来 提供 对 它 的 文 持 。 大 家 将 在 以 后 的 章节 里 详细 了 解 创 作 、 继 承 以 及 
它们 对 构建 费 造 成 的 影响 。 
































4.7 练习 

(1) 用 默认 构建 器 创建 一 个 类 (没有 自 变 量 ) ， 用 它 打 印 一 条 消息 。 创 
建 属于 这 个 类 的 一 个 对 象 。 

(2) 在 练习 1 的 基础 上 增加 一 个 过 载 的 构建 占 ， 令 其 采用 一 个 String 自 变 
量 ， 并 随同 自己 的 消息 打印 出 来 。 


(3) 以 练习 2 创建 的 类 为 基础 上 ， 创 建 属 于 它 的 对 象 句柄 的 一 个 数组 ， 但 
不 要 实际 创建 对 象 并 分 配 到 数组 里 。 运 行程 序 时 ， 注 意 是 否 打印 出 来 自 
构建 器 调用 的 初始 化 消 妃 。 


(4) 创建 同 句 柄 数组 联系 起 来 的 对 象 ， 最 终 完 成 练习 3。 
(5) 用 自 变 量 “before”，“after" 和 “none” 运 行程 序 ， 试 验 Garbage.java。 重 


复 这 个 操作 ， 观 察 是 否 从 输出 中 看 出 了 一 些 固定 的 模式 。 改 变 代码 ， 使 
System.runFinalization() 在 System.gc0O 之 前 调用 ， 再 观察 结果 。 





第 5 章 隐 闫 实施 过 程 


“进行 面 癌 对 象 的 设计 时 ， 一 项 基本 的 考 碟 是 : 如 何 将 发 生变 化 的 东西 
与 保持 不 变 的 东西 分 隔 开 。” 


这 一 点 对 于 库 来 说 是 特别 重要 的 。 那 个 库 的 用 户 《 客 户 程序 员 ) 必须 能 
依赖 自己 使 用 的 那 一 部 分 ， 并 知道 一 旦 新 版 本 的 库 出 台 ， 自 己 不 需要 改 
写 代 码 。 而 与 此 相反 ， 库 的 创建 者 必须 能 自由 地 进行 修改 与 改进 ， 同 时 
保证 客户 程序 员 代 码 不 会 受到 那些 变动 的 影响 。 


为 达到 这 个 目的 ， 需 遵守 一 定 的 约定 或 规则 。 例 如 ， 库 程序 员 在 修改 库 
内 的 一 个 类 时 ， 必 须 保证 不 删除 已 有 的 方法 ， 因 为 那样 做 会 造成 客户 程 
序 员 代 码 出 现 断 点 。 然 而 ， 相 到 的 情况 却 是 令 人 痛 否 的 。 对 于 一 个 数据 
成 员 ， 库 的 创建 者 怎样 才能 知道 哪些 数据 成 员 已 受到 客户 程序 员 的 访问 
呢 ? 知 方 法 属于 菜 个 类 唯一 的 一 部 分 ， 而 且 并 不 一 定 由 客户 程序 员 直 接 
使 用 ， 那 么 这 种 痛 天 的 情况 同样 是 真实 的 。 如 果 库 的 创建 者 想 删 除 一 种 
上 日 有 的 实施 方案 ， 并 置 入 新 代码 ， 此 时 又 该 怎么 办 呢 ? 对 那些 成 员 进 行 
的 任何 改动 都 可 能 中 断 客户 程序 员 的 代码 。 所 以 库 创 建 者 处 在 一 个 尴 粹 
的 境地 ， 似 乎 根本 动弹 不 得 。 


为 解决 这 个 问题 ，Java 推 出 了 “访问 指示 符 ” 的 概念 ， 人 允许 库 创建 者 声明 
哪些 东西 是 客户 程序 员 可 以 使 用 的 ， 哪 些 是 不 可 使 用 的 。 这 种 访问 控制 
的 级 别 在 “最 大 访问 ”和 “最 小 访问 ”的 范围 之 间 ， 分 别 包括 : public, “Ac 
好 的 ”( 无 关键 字 ) ，protected 以 及 private。 根 据 前 一 段 的 描述 ， 大 家 或 
许 已 总 结 出 作为 一 名 库 设计 者 ， 应 将 所 有 东西 都 尽 可 能 保持 

Aprivate” (AVA) ， 并 只 展示 出 那些 想 让 客户 程序 员 使 用 的 方法 。 这 
种 思路 是 完全 正确 的 ， 尽 管 它 有 点 儿 违 背 那 些 用 其 他 语言 (特别 是 C) 
编程 的 人 的 直觉 ， 那 些 人 习惯 于 在 没有 任何 限制 的 情况 下 访问 所 有 东 
西 。 到 这 一 童 结束 时 ， 大 家 应 该 可 以 深刻 体会 到 Java 访 问 控制 的 价值 。 


然而 ， 组 件 库 以 及 控制 谁 能 访问 那个 库 的 组 件 的 概念 现在 仍 不 是 完整 
的 。 仍 存在 这 样 一 个 问题 : 如 何 将 组 件 绑 定 到 单独 一 个 统一 的 库 单 元 
里 。 这 是 通过 Java 的 package GJE) 关键 字 来 实现 的 ， 而 且 访 问 指 示 符 
要 受到 类 在 相同 的 包 还 是 在 不 同 的 包 里 的 影响 。 所 以 在 本 章 的 开头 ， 大 
家 首先 要 学 习 库 组 件 如 何 置 入 包 里 。 这 样 才能 理解 访问 指示 符 的 完整 含 
NMa 












































5.1 包 : 库 单 元 


我 们 用 import 关 键 字 导入 一 个 完整 的 库 时 ， 就 会 获得 “ 包 ”(Package) 。 
例如 : 

import java.util.*; 

它 的 作用 是 导入 完整 的 实用 工具 〈Utility) 库 ， 该 库 属 于 标准 Java 开 发 
工具 包 的 一 部 分 。 由 于 Vector 位 于 java.uti 里 ， 所 以 现在 要 么 指定 完整 名 
称 “java.util.Vector”( 可 省 略 import 语 句 ) ， 要 么 简单 地 指定 一 

个 “Vector”( 因 为 import 是 默认 的 ) 。 


若 想 导入 单独 一 个 类 ， 可 在 import 语 句 里 指定 那个 类 的 名 字 : 








import java.util.Vector; 


现在 ， 我 们 可 以 自由 地 使 用 Vector。 然 而 ，java.util 中 的 其 他 任何 类 仍 是 
不 可 使 用 的 。 


之 所 以 要 进行 这 样 的 导入 ， 是 为 了 提供 一 种 特殊 的 机 制 ， 以 便 管 理 “ 命 

ZTE” (Name Space) 。 我 们 所 有 类 成 员 的 名 字 相 互 间 都 会 隔离 起 
来 。 位 于 类 A 内 的 一 个 方法 f0 不 会 与 位 于 类 B 内 的 、 拥 有 相同 “ 答 

名 ”( 自 变量 列表 ) 的 f0 发 生 冲 突 。 但 类 名 会 不 会 冲突 昵 ?” 假 设 创建 一 

个 stack 类 ， 将 它 安装 到 已 有 一 个 stack 类 (由 其 他 人 编写 ) 的 机 器 上 ， 这 
时 会 出 现 什么 情况 呢 ? 对 于 因特网 中 的 Java 心 用 ， 这 种 情况 会 在 用 户主 
不 知晓 的 时 候 发 生 ， 因 为 类 会 在 运行 一 个 Java 程 序 的 时 候 上 自动 下 载 。 


正 是 由 于 存在 名 字 潜 在 的 冲突 ， 所 以 特别 有 必要 对 Java 中 的 命名 空间 进 
行 完整 的 控制 ， 而 且 需 要 创建 一 个 完全 独一无二 的 名 字 ， 无 论 因特网 存 
在 什么 样 的 限制 。 


运 今 为 止 ， 本 书 的 大 多 数 例子 都 仅 存在 于 单个 文件 中 ， 而 且 设 计 成 局 部 
本地) 使 用 ， 没 有 同 包 名 发 生 冲 突 在 这 种 情况 下 ， 类 名 置 于 “默认 
WA) 。 这 是 一 种 有 效 的 做 法 ， 而 且 考 处 到 问题 的 简化 ， 本 书 剩 下 的 
部 分 也 将 尽 可 能 地 采用 它 。 然 而 ， 硅 计划 创建 一 个 “对 因特网 友好 ”或 者 
说 “适合 在 因特网 使 用 ”的 程序 ， 必 须 考 虑 如 何 防止 类 名 的 重复 。 

















为 Java 创 建 一 个 源码 文件 的 时 候 ， 它 通常 叫 作 一 个 “编辑 单元 ”( 有 时 也 
叫 作 “翻译 单元 ”>) 。 每 个 编译 单元 都 必须 有 一 个 以 ,java 结尾 的 名 字 。 而 
且 在 编译 单元 的 内 部 ， 可 以 有 一 个 公共 〈public) 类 ， 它 必须 拥有 与 文 
件 相同 的 名 字 包括 大 小 写 形式 ， 但 排除 .java 文 件 扩展 名 ) 。 如 果 不 这 
样 做 ， 编 译 占 就 会 报告 出 错 。 每 个 编译 单元 内 都 只 能 有 一 个 public 类 
(同样 地 ， 否 则 编译 器 会 报告 出 错 ) 。 那 个 编译 单元 剩 下 的 类 〈 如 果 有 
的 话 ) 可 在 那个 包 外 面 的 世界 面前 隐藏 起 来 ， 因 为 它们 并 非 “ 公 共 ? 的 
( 非 public〉， 而 且 它 们 由 用 于 主 public 类 的 “支撑 ”类 组 成 。 


编译 一 个 .java 文 件 时 ， 我 们 会 获得 一 个 名 字 完 全 相同 的 输出 文件 ， 但 对 
于 .java 文 件 中 的 每 个 类 ， 它 们 都 有 一 个 .class 扩 展 名 。 因 此 ， 我 们 最 终 从 
少量 的 .java 文 件 里 有 可 能 获得 数量 众多 的 .class 文 件 。 如 以 前 用 一 种 汇编 
语言 写 过 程序 ， 那 么 可 能 已 习惯 编译 占 先 分 割 出 一 种 过 渡 形 式 〈( 通 常 是 
一 个 .obj 文 件 ) ， 再 用 一 个 链接 器 将 其 与 其 他 东西 封装 到 一 起 〈 生 成 一 
个 可 执行 文件 ) ， 或 者 与 一 个 库 封 装 到 一 起 〈 生 成 一 个 库 ) 。 但 那 并 不 
是 Java 的 工作 方式 。 一 个 有 效 的 程序 就 是 一 系列 .class 文 件 ， 它 们 可 以 封 
装 和 压缩 到 一 个 JAR 文 件 里 《使 用 Java 1.1 提 供 的 jar 工 具 ) 。Java 解 释 器 
负责 对 这 些 文件 的 寻找 、 装 载 和 解释 GERO). 

O: Java 并 没有 强制 一 定 要 使 用 解释 器 。 一 些 回 有 代码 的 Java 编 译 器 可 
生成 单独 的 可 执行 文件 。 


“ 库 ” 也 由 一 系列 类 文件 构成 。 每 个 文件 都 有 一 个 public 类 (并 没 强 迫使 

用 一 个 public 类 ， 但 这 种 情况 最 很 典型 的 ) ， 所 以 每 个 文件 都 有 一 个 组 

件 。 如 果 想 将 所 有 这 些 组 件 ( 它 们 在 各 目 独 立 的 .java 和 .class 文 件 里 ) 都 
归纳 到 一 起 ， 那 么 package 关 键 字 就 可 以 发 挥 作 用 ) 。 


在 在 一 个 文件 的 开头 使 用 下 述 代码 : 























package mypackage; 


那么 package 语 句 必 须 作 为 文件 的 第 一 个 非 注 释 语句 出 现 。 该 语句 的 作 
用 是 指出 这 个 编译 单元 属于 名 为 mypackage 的 一 个 库 的 一 部 分 。 或 者 换 
人 句 话 说 ， 它 表明 这 个 编译 单元 内 的 public 类 名 位 于 mypackage 这 个 名 字 的 
下 面 。 如 果 其 他 人 想 使 用 这 个 名 字 ， 要 么 指出 完整 的 名 字 ， 要 么 与 
mypackage 联 合 使 用 import 关 键 字 (使 用 前 面 给 出 的 选项 ) 。 注 意 根据 
Java, HE) 的 约定 ， 名 字 内 的 所 有 字母 都 应 小 写 ， 甚 至 那些 中 间 单 
词 亦 要 如 此 。 








例如 ， 假 定 文件 名 是 MyClass.java。 它 意味 着 在 那个 文件 有 一 个 、 而 且 
只 能 有 一 个 public 类 。 而 且 那 个 类 的 名 字 必 须 是 MyClass (包括 大 小 写 形 
cave ae 


package mypackage; 
public class MyClass { 
//... 


现在 ， 如 果 有 人 想 使 用 MyClass， 或 者 想 使 用 mypackage 内 的 其 他 任何 
public 类 ， 他 们 必须 用 import 关 键 字 激活 mypackage 内 的 名 字 ， 使 它们 能 
够 使 用 。 男 一 个 办 法 则 是 指定 完整 的 名 称 : 


mypackage.MyClass m = new mypackage.MyClass(); 
import 关 键 字 则 可 将 其 变 得 简洁 得 多 : 

import mypackage.*; 

EnA 


MyClass m = new MyClass(); 


作为 一 名 库 设计 者 ， 一 定 要 记 住 package 和 import 关 键 字 允许 我 们 做 的 事 
情 就 是 分 割 单个 全 局 命名 空间 ， 保 证 我 们 不 会 遇 到 名 字 的 冲突 一 一 无 论 
有 多 少 人 使 用 因特网 ， 也 无 论 多 少 人 用 Java 编 写 自己 的 类 。 


5.1.1 创建 独一无二 的 包 名 


大 家 或 许 已 注意 到 这 样 一 个 事实 : 由 于 一 个 包 永 远 不 会 真 的 “ 封 沪 ”到 蛙 
独 一 个 文件 里 面 ， 它 可 由 多 个 .class 文 件 构成 ， 所 以 局 面 可 能 稍微 有 些 混 
乱 。 为 避免 这 个 问题 ， 最 合理 的 一 种 做 法 就 是 将 某 个 特定 包 使 用 的 所 
有 .class 文 件 部 置 入 单个 目录 里 。 也 就 是 说 ， 我 们 要 利用 操作 系统 的 分 级 
文件 结构 避免 出 现 混乱 局 面 。 这 正 是 Java 所 采取 的 方法 。 


它 同时 也 解决 了 另 两 个 问题 : 创建 独一无二 的 包 名 以 及 找 出 那些 可 能 深 
北 于 目录 结构 某 处 的 类 。 正 如 我 们 在 第 2 章 讲 述 的 那样 ， 为 达到 这 个 目 
的 ， 需 要 将 .class 文 件 的 位 置 路 径 编 码 到 package 的 名 字 里 。 但 根据 约 








定 ， 编 译 器 强迫 package 名 的 第 一 部 分 是 类 创建 者 的 因特网 域名 。 由 于 
因特网 域名 肯定 是 独一无二 的 〈 由 InterNIC 保 证 一 ”注释 @， 它 控制 着 
域名 的 分 配 ) ， 所 以 假如 按 这 一 约定 行事 ，package 的 名 称 就 肯定 不 会 
重复 ， 所 以 永远 不 会 遇 到 名 称 冲突 的 问题 。 换 名 话说 ， 除 非 将 自己 的 域 
名 转让 给 其 他 人 ， 而 且 对 方 也 按照 相同 的 路 径 名 编写 Java 代 码 ， 否 则 名 
字 的 冲突 是 永远 不 会 出 现 的 。 当 然 ， 如 果 你 没有 自己 的 域名 ， 那 么 必须 
创造 一 个 非常 生僻 的 包 名 《例如 自己 的 英文 姓名 ) ， 以 便 尽 最 大 可 能 创 
建 一 个 独一无二 的 包 名 。 如 决定 发 行 上 自己 的 Java 人 代码， 那么 强烈 推荐 去 
申请 自己 的 域名 ， 它 所 需 的 费用 是 非常 低廉 的 。 


@): ftp://ftp.internic.net 


这 个 技巧 的 男 一 部 分 是 将 package 名 解析 成 自己 机 右上 的 一 个 日 录 。 这 
样 一 来 ，Java 程 序 运 行 并 需要 装载 .class 文 件 的 时 候 (这 是 动态 进行 的 ， 
在 程序 需要 创建 属于 那个 类 的 一 个 对 象 ， 或 者 首次 访问 那个 类 的 一 个 
static 成 员 时 ) ， 它 就 可 以 找到 .class 文 件 驻 留 的 那个 目录 。 


Java 解 释 器 的 工作 程序 如 下 : 首先 ， 它 找到 环境 变量 CLASSPATH (将 
Java 或 者 具有 Java 解 释 能 力 的 工具 一 一 如 浏览 器 一 一 安装 到 机 器 中 时 ， 
通过 操作 系统 进行 设 定 ) 。CLASSPATH 包 含 了 一 个 或 多 个 目录 ， 它 们 
作为 一 种 特殊 的 “ 根 ” 使 用 ， 从 这 里 展开 对 .class 文 件 的 搜索 。 从 那个 根 开 
始 ， 解 释 器 会 寻找 包 名 ， 并 将 每 个 点 写 〈( 句 点 ) 葵 换 成 一 个 和 斜 枉 ， 从 而 
生成 从 CLASSPATH 根 开始 的 一 个 路 径 名 〈 上 所 以 package foo.bar.baz 会 变 
成 foo\banbaz 或 者 foo/barvbaz; 具体 是 正 斜 杠 还 是 反 斜 杠 由 操作 系统 决 
定 ) 。 随 后 将 它们 连接 到 一 起 ， 成 为 CLASSPATH 内 的 各 个 条 目 〈 入 
口 ) 。 以 后 搜索 .class 文 件 时 ， 束 可 从 这 些 地 方 开 始 碍 找 与 准备 创建 的 类 
名 对 应 的 名 字 。 此 外 ， 它 也 会 搜索 一 些 标准 目录 这 些 目录 与 Java 解 
释 器 驻 留 的 地 方 有 关 。 


为 进一步 理解 这 个 问题 ， 下 面 以 我 自己 的 域名 为 例 ， 它 是 
bruceeckel.com。 将 其 反 转 过 来 后 ，com.bruceeckel 就 为 我 的 类 创建 了 独 
一 无 二 的 全 局 名 称 (com，edu，org，net 等 扩展 名 以 前 在 Java 包 中 都 是 
ASW, (ABJava ”1.2 以 来 ， 这 种 情况 已 发 生 了 变化 。 现 在 整个 包 名 都 
是 小 写 的 ) 。 由 于 决定 创建 一 个 名 为 util 的 库 ， 我 可 以 进一步 地 分 割 
它 ， 所 以 最 后 得 到 的 包 名 如 下 : 























package com.bruceeckel.util; 


现在 ， 可 将 这 个 包 名 作为 下 述 两 个 文件 的 “命名 空间 ”使 用 : 


//: Vector.java 


// Creating a package 
package com.bruceeckel.util; 
public class Vector { 
public Vector() { 
System. out.printin( 
"com.bruceeckel.util.Vector"); 


} 
A f/f 


创建 自己 的 包 时 ， 要 求 package 语 句 必 须 是 文件 中 的 第 一 个 “ 非 注释 ” 代 
码 。 第 二 个 文件 表面 看 起 来 是 类 似 的 : 


//: List.java 


// Creating a package 
package com.bruceeckel.util; 
public class List { 
public List() { 
System.out.printin( 


"com.bruceeckel.util.List"); 


} ///:~ 


这 两 个 文件 都 置 于 我 自己 系统 的 一 个 子 目 录 中 : 


C:\DOC\JavaT\com\bruceeckel\util 





通过 它 往 回 走 ， 就 会 发 现 包 名 com.bruceeckel.utilj， 但 路 径 的 第 一 部 分 
是 什么 昵 ? 这 是 由 CLASSPATH 环 境 变 量 决 定 的 。 在 我 的 机 器 上 ， 它 


aim Yl AE 


CLASSPATH=.;D:\JAVA\LIB;C:\DOCVavaT 


可 以 看 出 ，CLASSPATH 里 能 包含 大 量 备用 的 搜索 路 径 。 然 而 ， 使 用 
JAR 文 件 时 要 注意 一 个 问题 : 必须 将 JAR 文 件 的 名 字 置 于 类 路 径 里 ， 而 
不 仅仅 是 它 所 在 的 路 径 。 所 以 对 一 个 名 为 grape.jar 的 JAR 文 件 来 说 ， 我 
们 的 类 路 径 需 要 包括 : 
CLASSPATH=.;D:JAVA\LIB;C:\flavors\grape.jar 


正确 设置 好 类 路 径 后 ， 可 将 下 面 这 个 文件 置 于 任何 目录 里 《〈 知 在 执行 该 
程序 时 过 到 麻烦 ， 请 参见 第 3 半 的 3.1.2 小 市 “赋值 ”): 


//: LibTest.java 





// Uses the library 
package c05; 
import com.bruceeckel.util.*; 
public class LibTest { 
public static void main(String[] args) { 
Vector v = new Vector(); 


List 1 = new List(); 


} ///:~ 


编译 器 过 到 import 语 句 后 ， 它 会 搜索 由 CLASSPATH 指 定 的 目录 ， 查 找 
子 目 录 com\bruceeckel\util， 然 后 查找 名 称 适 当 的 已 编译 文件 (对 于 
Vector 是 Vector.class， 对 于 List 则 是 List.class 〉。 注 意 Vector 和 List 内 无 
论 类 还 是 需要 的 方法 都 必须 设 为 public。 


1. 自动 编译 


为 导入 的 类 首次 创建 一 个 对 象 时 (或 者 访问 一 个 类 的 static 成 员 时 ) ， 编 
译 器 会 在 适当 的 目录 里 寻找 同名 的 .class 文 件 〈 所 以 如 果 创 建 类 X 的 一 个 
对 象 ， 束 应 该 是 X.class) 。 若 只 发 现 X.class， 它 就 是 必须 使 用 的 那 一 个 
类 。 人 然而， 如 果 它 在 相同 的 目录 中 还 发 现 了 一 个 X.java， 编 译 占 就 会 比 
较 两 个 文件 的 日 期 标记 。 如 果 X.java 比 X.class 新 ， 就 会 自动 编译 X.java， 
生成 一 个 最 新 的 X.class。 


对 于 一 个 特定 的 类 ， 或 在 与 它 同 名 的 .java 文 件 中 没有 找到 它 ， 束 会 对 那 
个 类 采取 上 述 的 处 理 。 


2. 冲突 


各 通过 * 导 入 了 两 个 库 ， 而 且 它们 包括 相同 的 名 字 ， 这 时 会 出 现 什么 情 
况 呢 ?例如 ， 假 定 一 个 程序 使 用 了 下 述 导 入 语句 : 


























import com.bruceeckel.util.*; 

import java.util.*; 

由 于 java.util.* 也 包含 了 一 个 Vector 类 ， 所 以 这 会 造成 潜在 的 冲突 。 然 
而 ， 只 要 冲突 并 不 真 的 有 发生， 那么 就 不 会 产生 任何 问题 一 一 这 当然 古 最 
理想 的 情况 ， 因 为 否则 的 话 ， 就 需要 进行 大 量 编程 工作 ， 防 范 那 些 可 能 
可 能 永远 也 不 会 友 生 的 冲突 。 


如 现在 试 着 生成 一 个 Vector， 就 肯定 会 及 生 冲 突 。 如 下 所 示 : 











Vector v = new Vector(); 


它 引 用 的 到 底 是 哪个 Vector 类 呢 ? 编译 器 对 这 个 问题 没有 答案 ， 读 者 也 








不 可 能 知道 。 所 以 编译 器 会 报告 一 个 错误 ， 强 迫 我 们 进行 明确 的 说 明 。 
例如 ， 假 设 我 想 使 用 标准 的 Java Vector， 那 么 必须 象 下 面 这 样 编程 : 


java.util. Vector v = new java.util. Vector(); 


由 于 它 (与 CLASSPATH 一 起 ) 完整 指定 了 那个 Vector 的 位 置 ， 所 以 不 
再 需要 import java.util.* 语 句 ， 除 非 还 想 使 用 来 自 java.util 的 其 他 东西 。 


5.1.2 自 定义 工具 库 

掌握 前 述 的 知识 后 ， 接 下 来 就 可 以 开始 创建 自己 的 工具 库 ， 以 便 减 少 或 
者 完全 消除 重复 的 代码 。 例 如 ， 可 为 System.out.println() 创 建 一 个 别名 ， 
减少 重复 键入 的 代码 量 。 它 可 以 是 名 为 tools 的 一 个 包 (package) 的 一 


部 分 ， 


//: P.java 





// The P.rint & P.rintln shorthand 
package com.bruceeckel.tools; 
public class P { 
public static void rint(Object obj) { 
System.out.print(obj); 
} 
public static void rint(String s) { 
System.out.print(s); 
} 
public static void rint(char[] s) { 


System.out.print(s); 


public static void rint(char c) { 
System.out.print(c); 

} 

public static void rint(int i) { 
System.out.print(i); 

J 

public static void rint(long 1) { 
System.out.print(1l); 

} 

public static void rint(float f) { 
System.out.print(f); 

Í 

public static void rint(double d) { 
System.out.print(d); 

I 

public static void rint(boolean b) { 
System.out.print(b); 

iF 

public static void rintln() { 
System.out.println(); 

} 

public static void rintln(Object obj) { 


System.out.println(obj); 


} 


public static void rintln(String s) { 
System.out.println(s); 

} 

public static void rintln(char[] s) { 
System.out.println(s); 

} 

public static void rintln(char c) { 
System.out.println(c); 

} 

public static void rintln(int i) { 
System.out.println(i); 

} 

public static void rintln(long 1) { 
System.out.println(1); 

} 

public static void rintln(float f) { 
System.out.println(f); 

} 

public static void rintln(double d) { 
System.out.println(d); 


} 


public static void rintln(boolean b) { 


System.out.println(b); 


} 
FA i 


所 有 不 同 的 数据 类 型 现在 都 可 以 在 一 个 新 行 输出 〈P.rinttn0) ， 或 者 不 
在 一 个 新 行 输出 〈P.rintO) 。 


大 家 可 能 会 猜想 这 个 文件 所 在 的 目录 必须 从 某 个 CLASSPATH 位 置 开 
始 ， 然 后 继续 com/bruceeckel/tools。 编 译 完毕 后 ， 利 用 一 个 import 语 
句 ， 即 可 在 自己 系统 的 任何 地 方 使 用 P.class 文 件 。 如 下 所 示 : 


ToolTest.java 


所 以 从 现在 开始 ， 无 论 什 么 时 候 只 要 做 出 了 一 个 有 用 的 新 工具 ， 束 可 将 
其 加 入 tools 目 录 或 者 上 自己 的 个 人 util 或 tools 目 录 ) 。 


1. CLASSPATH 的 陷阱 


P.java 文 件 存在 一 个 非常 有 趣 的 陷阱 。 特 别 是 对 于 早期 的 Java 实 现 方案 
来 说 ， 类 路 径 的 正确 设 定 通 党 都 是 很 困难 的 一 项 工作 。 编 写 这 本 书 的 时 
候 ， 我 引入 了 P.java 文 件 ， 它 最 初 看 起 来 似乎 工作 很 正常 。 但 在 茶 些 情 
况 下 ， 却 开始 出 现 中 断 。 在 很 长 的 时 间 里 ， 我 都 确信 这 是 Java 或 其 他 什 
么 在 实现 时 一 个 错误 。 但 最 后 ， 我 终于 发 现在 一 个 地 方 引 入 了 一 个 程序 
〈 即 第 17 章 要 说 明 的 CodePackager.java) ， 它 使 用 了 一 个 不 同 的 类 P。 

由 于 它 作为 一 个 工具 使 用 ， 所 以 有 时 候 会 进入 类 路 径 里 ， 另 一 些 时 候 则 
不 会 这 样 。 但 只 要 它 进 入 类 路 径 ， 那 么 假 知 执行 的 程序 需要 寻找 
com.bruceeckel.tools 中 的 类 ，Java 首 先 发 现 的 就 是 CodePackager.java 中 的 
P。 此 时 ， 编 译 器 会 报告 一 个 特定 的 方法 没有 找到 。 这 当然 是 非常 令 人 
头疼 的 ， 因 为 我 们 在 前 面 的 类 P 里 明明 看 到 了 这 个 方法 ， 而 且 根 本 没有 
更 多 的 诊断 报告 可 为 我 们 提供 一 条 线索 ， 让 我 们 知道 找到 的 是 一 个 完 
不 同 的 类 《〈 那 甚至 不 是 public 的 ) 。 


乍 一 看 来 ， 这 似乎 是 编译 需 的 一 个 错误 ， 但 假 奋 考察 import 语 句 ， 就 会 
发 现 它 只 是 说 :“ 在 这 里 可 能 发 现 了 P”。 然 和 而， 我们 假定 的 是 编译 圳 搜 
索 自 己 类 路 径 的 任何 地 方 ， 所 以 一 旦 它 发 现 一 个 P， 就 会 使 用 它 ; FE 
搜索 过 程 中 发 现 了 “错误 的 ”一 个 ， 它 就 会 停止 搜索 。 这 与 我 们 在 前 面 表 























述 的 稍微 有 些 区 别 ， 因 为 存在 一 些 讨厌 的 类， 它们 都 位 于 包 内 。 而 这 里 
有 一 个 不 在 包 内 的 Pp， 但 仍 可 在 常规 的 类 路 径 搜 索 过 程 中 找到 。 


如 果 您 遇 到 象 这 样 的 情况 ， 请 务必 保证 对 于 类 路 径 的 每 个 地 方 ， 每 个 名 
字 都 仅 存在 一 个 类 。 


5.1.3 利用 导入 改变 行为 


Java 已 取消 的 一 种 特性 是 C 的 “条 件 编译 ”， 它 允许 我 们 改变 参数 ， 获 得 
不 同 的 行为 ， 同 时 不 改变 其 他 任何 代码 。Java 之 所 以 抛弃 了 这 一 特性 ， 
可 能 是 由 于 该 特性 经 常 在 C 里 用 于 解决 跨 平 台 问 题 ， 代 人 码 的 不 同 部 分 根 
据 具 体 的 平台 进行 编译 ， 人 否则 不 能 在 特定 的 平台 上 运行 。 由 于 Java 的 设 
计 思 想 是 成 为 一 种 自动 跨 平 台 的 语言 ， 所 以 这 种 特性 是 没有 必要 的 。 


然而 ， 条 件 编译 还 有 男 一 些 非常 有 价值 的 用 途 。 一 种 很 常见 的 用 途 束 是 
调试 代码 。 调 试 特性 可 在 开发 过 程 中 使 用 ， 但 在 发 行 的 产品 中 却 无 此 功 
能 。Alen Holub (www.holub.com) 提出 了 利用 包 〈package) 来 模仿 条 
件 编 译 的 概念 。 根 据 这 一 概念 ， 它 创建 了 C“ 断 定 机 制 ” 一 个 非常 有 用 的 
Java 厂 本。 之 所 以 叫 作 “断定 机 制 ?”， 是 由 于 我 们 可 以 说 “ 它 应 该 为 真 ? 或 
者 “ 它 应 该 为 假 "?。 如 果 语 句 不 同意 你 的 断定 ， 束 可 以 发 现 相关 的 情况 。 
这 种 工具 在 调试 过 程 中 是 特别 有 用 的 。 


可 用 下 面 这 个 类 进行 程序 调试 : 


//: Assert.java 

















// Assertion tool for debugging 
package com.bruceeckel.tools.debug; 
public class Assert { 
private static void perr(String msg) { 
System.err.println(msg); 
} 


public final static void is true(boolean exp) { 


if(!exp) perr("Assertion failed"); 
} 
public final static void is false(boolean exp)t{ 
if(exp) perr("Assertion failed"); 
} 
public final static void 
is_true(boolean exp, String msg) { 
if(!exp) perr("Assertion failed: " + msg); 
} 
public final static void 
is_false(boolean exp, String msg) { 
if(exp) perr("Assertion failed: " + msg); 
} 
-XA 


这 -1 TAN Sl 如 果 失 败 ， 就 显示 出 出 错 消息 。 在 
第 9 章 ， 大 家 会 学 习 一 个 更 高 级 的 错误 控制 工具 ， 名 为 “违例 控制 ”。 
但 在 目前 T 情况 下 ，perr() 方 法 已 经 可 以 很 好 地 工作 。 


如 果 想 使 用 这 个 类 ， 可 在 目 己 的 程序 中 加 入 下 面 这 一 行 : 








import com.bruceeckel.tools.debug.*; 


如 欲 清除 断定 机 制 ， 以 便 上 自己 能 发 行 最 终 的 代码 ， 我 们 创建 了 第 二 个 
Assert 类 ， 但 却 是 在 一 个 不 同 的 包 里 : 


//: Assert.java 





// Turning off the assertion output 

// so you can ship the program. 

package com.bruceeckel.tools; 

public class Assert { 
public final static void is_true(boolean exp) {} 
public final static void is_false(boolean exp) {} 
public final static void 
is_true(boolean exp, String msg) {} 
public final static void 
is_false(boolean exp, String msg) {} 


Af] fv 


现在 ， 假 如 将 前 一 个 import 语 句 变 成 下 面 这 个 样子 : 
import com.bruceeckel.tools.*; 


程序 便 不 再 显示 出 断言 。 下 面 是 个 例子 : 


//: TestAssert.java 





// Demonstrating the assertion tool 

package c05; 

// Comment the following, and uncomment the 

// subsequent line to change assertion behavior: 
import com.bruceeckel.tools.debug.*; 


// import com.bruceeckel.tools.*; 


public class TestAssert { 
public static void main(String[] args) { 
Assert.is_true((2 + 2) == 5); 
Assert.is_ false((1 + 1) == 2); 
Assert.is_true((2 + 2) == 5, "2 + 2 == 5"); 
Assert.is_false((1 + 1) == 2, "1 +1 != 2"); 
} 
} ///:~ 





通过 改变 导入 的 package， 我 们 可 将 目 己 的 代码 从 调试 版 本 变 成 最 终 的 
发 行 版 本 。 这 种 技术 可 应 用 于 任何 种 类 的 条 件 代码 。 


5.1.4 包 的 停 用 


大 家 应 注意 这 样 一 个 问题 ， 每 次 创建 一 个 包 后 ， 都 在 为 包 取 名 时 间接 地 
指定 了 一 个 目录 结构 。 这 个 包 必 须 存在 〈 驻 留 ) 于 由 它 的 名 字 规 定 的 目 
录 内 。 而 且 这 个 目录 必须 能 从 CLASSPATH 开 始 搜索 并 发 现 。 最 开始 的 
时 候 ，package 关 键 字 的 运用 可 能 会 令 人 迷惑 ， 因 为 除非 坚持 遵守 根据 
目录 路 径 指 定 包 名 的 规则 ， 和 否则 就 会 在 运行 期 获得 大 量 莫 名 其 妙 的 消 
妃 ， 指 出 找 不 到 一 个 特定 的 类 即使 那个 类 明明 就 在 相同 的 目录 中 。 
若 得 到 象 这 样 的 一 条 消息 ， 请 试 着 将 package 语 句 作 为 注释 标记 出 去 。 
如 果 这 样 做 行 得 通 ， 融 可 知道 问题 到 底 出 在 哪儿 。 























5.2 Java 访 问 指示 符 


针对 类 内 每 个 成 员 的 每 个 定义 ，Java 访 问 指示 符 poublic，protected 以 及 
private 都 置 于 它们 的 最 前 面 一 一 无 论 它 们 是 一 个 数据 成 员 ， 还 是 一 个 方 
法 。 每 个 访问 指示 符 都 只 控制 着 对 那个 特定 定义 的 访问 。 这 与 C++ 存在 
者 显著 不 同 。 在 C++ 中 ， 访 问 指示 符 控 制 着 它 后 面 的 所 有 和 定义， 直到 又 
一 个 访问 指示 符 加 入 为 止 。 


通过 干 丝 万 缕 的 联系 ， 程 序 为 所 有 东西 部 指定 了 茶 种 形式 的 访问 。 在 后 
a 大 家 要 学 习 与 各 类 访问 有 关 的 所 有 知识 。 首 次 从 默认 访问 
开始 。 


5.2.1“ 友 好 的 ” 


如 打 根 本 不 指定 访问 指示 符 ， 残 象 本 章 之 前 的 所 有 例子 那样 ， 这 时 会 出 
现 什 么 情况 呢 ? 默认 的 访问 没有 关键 字 ， 但 它 通常 称 为 " 友 

ue” (Friendly) 访问 。 这 意味 着 当前 包 内 的 其 他 所 有 类 都 能 访问 “友好 

的 ?成员 ， 但 对 包 外 的 所 有 类 来 说 ， 这 些 成 员 却 是 “私有 ”(Private) 的 ， 

外 界 不 得 访问 。 由 于 一 个 编译 单元 〈 一 个 文件 ) 只 能 从 属于 单个 包 ， 所 
以 单个 编译 单元 内 的 所 有 类 相互 间 都 是 目 动 * 友 好 ”的 。 因 此， 我 们 也 说 
友好 元 素 拥 有 “ 包 访 问 * 权 限 。 


友好 访问 允许 我 们 将 相关 的 类 都 组 合 到 一 个 包 里 ， 使 它们 相互 间 方 便 地 
进行 沟通 。 将 类 组 合 到 一 个 包 内 以 后 这样 便 允许 友好 成 员 的 相互 访 

问 ， 亦 即 让 它们“* 交 朋友 ”) ， 我 们 便 “ 拥 有 ”了 那个 包 内 的 代码 。 只 有 我 
们 已 经 拥有 的 代码 才能 友好 地 访问 自己 拥有 的 其 他 代码 。 我 们 可 认为 友 
好 访问 使 类 在 一 个 包 内 的 组 合 显 得 有 意义 ， 或 者 说 前 者 是 后 者 的 原因 。 
在 许多 语言 中 ， 我 们 在 文件 内 组 织 定 义 的 方式 往往 显得 有 些 牵 强 。 但 在 
Java 中 ， 却 强制 用 一 种 颇 有 意义 的 形式 进行 组 织 。 除 此 以 外 ， 我 们 有 时 
可 能 想 排 除 一 些 类 ， 不 想 让 它们 访问 当前 包 内 定义 的 类 。 


对 于 任何 关系 ， 一 个 非常 重要 的 问题 是 “ 谁 能 访问 我 们 的 ‘私有 ’ 或 private 
代码 ”。 类 控制 着 哪些 代码 能 够 访问 目 己 的 成 员 。 没 有 任何 秘诀 可 以 “加 
Dr. AP ELA HEE LRH ars, PAR: e, FQAEBobAY HA 
友 ! ”， 并 指望 看 到 Bob 的 “protected” (受到 保护 的 ) 、 友 好 的 以 

及 “private”( 私 有 ) 的 成 员 。 为 获得 对 一 个 访问 权限 ， 唯 一 的 方法 整 





























H 
XE: 


(1) 使 成 员 成 为 “public”( 公 共 的 ) 。 这 样 所 有 人 从 任何 地 方 都 可 以 访问 
Eo 





D ” 变 成 一 个 “友好 ”成 员 ， 方 法 是 舍弃 所 有 访问 指示 符 ， 并 将 其 类 置 于 
相同 的 包 内 。 这 样 一 来 ， 其 他 类 就 可 以 访问 成 员 。 


(B) 正如 以 后 引入 “继承 ”概念 后 大 家 会 知道 的 那样 ， 一 个 继承 的 类 既 可 
以 访问 一 个 protected 成 员 ， 也 可 以 访问 一 个 public 成 员 《〈 但 不 可 访问 
private 成 员 ) 。 只 有 在 两 个 类 位 于 相同 的 包 内 时 ， 它 才 可 以 访问 友好 成 
员 。 但 现在 不 必 关 心 这 方面 的 问题 。 


(4) 提供 “访问 器 / 变化 器 "方法 〈 亦 称 为 “获取 / 设置 ”方法 ) ， 以 便 读 取 
和 修改 值 。 这 是 OOP 环 境 中 最 正规 的 一 种 方法 ， 也 是 Java ” ”Beans 的 基础 
具体 情况 会 在 第 13 章 介绍 。 





5.2.2 public: 接口 访问 

使 用 public 关 键 字 时 ， 它 意味 着 紧 随 在 public 后 面 的 成 员 声 明 适 用 于 所 有 
人 ， 特 别 是 适用 于 使 用 库 的 客户 程序 员 。 假 定 我 们 定义 了 一 个 名 为 
dessert 的 包 ， 其 中 包含 下 述 单元 〈 若 执行 该 程序 时 遇 到 困难 ， 请 参考 第 
3 章 3.1.2 小 节 “ 赋 值 >) : 


//: Cookie.java 


// Creates a library 
package c05.dessert; 
public class Cookie { 
public Cookie() { 
System.out.printin( "Cookie constructor"); 


} 


void foo() { System.out.println("foo"); } 


} ///:~ 


请 记 住 ，Cookie.java 必 须 驻 留 在 名 为 dessert 的 一 个 子 目 录 内 ， 而 这 个 子 
目录 又 必须 位 于 由 CLASSPATH 指 定 的 C05 目 录 下 面 〈C05 代 表 本 书 的 第 
5 章 ) 。 不 要 错误 地 以 为 Java 无 论 如 何 都 会 将 当前 目录 作为 搜索 的 起 点 
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现在 ， 假 车 创建 使 用 了 Cookie 的 一 个 程序 ， 如 下 所 示 : 


//: Dinner.java 





// Uses the library 
import c05.dessert.*; 
public class Dinner { 
public Dinner() { 
System.out.printin("Dinner constructor"); 
} 
public static void main(String[] args) { 
Cookie x = new Cookie(); 
//! x.foo(); // Can't access 
} 
二 


就 可 以 创建 一 个 Cookie 对 象 ， 因 为 它 的 构建 器 是 public 的 ， 而 且 类 也 是 
public 的 《公共 类 的 概念 稍 后 还 会 进行 更 详细 的 讲述 ) 。 然 而 ，foo0) 成 
员 不 可 在 Dinner.java 内 访问 ， 因 为 foo0 只 有 在 dessert 包 内 才 是 “ 友 

好 ”的 。 


1. 默认 包 


大 家 可 能 会 惊讶 地 发 现下 面 这 文 些 代 码 得 以 顺利 编译 一 一 尽管 它 看 起 来 似 
乎 已 违背 了 规则 : 


//: Cake.java 


// Accesses a class in a separate 
// compilation unit. 
class Cake { 
public static void main(String[] args) { 
Pie x = new Pie(); 
x.F(); 
} 
} ///:~ 


在 位 于 相同 目录 的 第 二 个 文件 里 : 


//: Pie.java 


// The other class 
class Pie { 
void f() { System.out.printin("Pie.f()"); } 


} ///:~ 


最 初 可 能 会 把 它们 看 作 完全 不 相干 的 文件 ， 然 而 Cake 能 创建 一 个 Pie 对 
象 ， 并 能 调用 它 的 {0 方法 ! 通常 的 想法 会 认为 Pie 和 {0 是 “友好 的 "， 所 以 
不 适用 于 Cake。 它 人 部 分 结论 非常 正确 。 但 它们 之 
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所 以 仍 能 在 Cake.java 中 使 用 ， 是 由 于 它们 位 于 相同 的 目录 中 ， 而 且 没 有 
明确 的 包 名 。Java 把 象 这 样 的 文件 看 作 那 个 目录 “默认 包 ” 的 一 部 分 ， 所 
以 它们 对 于 目录 内 的 其 他 文件 来 说 是 “友好 ”的 。 


5.2.3 private: 不 能 接触 ! 


private 关 键 字 意味 着 除非 那个 特定 的 类 ， 而 且 从 那个 类 的 方法 里 ， 否 则 
没有 人 能 访问 那个 成 员 。 同 一 个 包 内 的 其 他 成 员 不 能 访问 private 成 员 ， 

这 使 其 显得 似乎 将 类 与 我 们 自己 都 隔离 起 来 。 男 一 方面 ， 也 不 能 由 几 个 
合作 的 人 创建 一 个 包 。 所 以 private 允 许 我 们 自由 地 改变 那个 成 员 ， 同 时 
组 需 关 心 它 是 否 会 影响 同一 个 包 内 的 另 一 个 类 。 默 认 的 “友好 ”" 包 访问 通 
常 已 经 是 一 种 适当 的 隐藏 方法 ， 请 记 住 ， 对 于 包 的 用 户 来 说 ， 是 不 能 访 
问 一 个 “友好 ?成 员 的 。 这 种 效果 往往 能 令 人 满意 ， 因 为 默认 访问 是 我 们 
通常 采用 的 方法 。 对 于 希望 变 成 public (AR) 的 成 员 ， 我 们 通常 明确 
地 指出 ， 令 其 可 由 客户 程序 员 自 由 调用 。 而 且 作 为 一 个 结果 ， 最 开始 的 
时 候 通 常会 认为 自己 不 必 频 繁 使 用 private 关 键 字 ， 因 为 完全 可 以 在 不 用 
它 的 前 提 下 发 布 自己 的 代码 〈 这 与 C++ 是 个 鲜明 的 对 比 ) 。 然 而 ， 随 着 
学 习 的 深入 ， 大 家 就 会 发 现 private 仍 然 有 非常 重要 的 用 途 ， 特 别 是 在 涉 
及 多 线程 处 理 的 时 候 (详情 见 第 14 章 )〉。 


下 面 是 应 用 了 private 的 一 个 例子 : 


//: IceCream.java 





























// Demonstrates "private" keyword 
class Sundae { 

private Sundae() {} 

static Sundae makeASundae() { 


return new Sundae(); 


} 


public class IceCream { 


public static void main(String[] args) { 
//! Sundae x = new Sundae(); 
Sundae x = Sundae.makeASundae(); 

} 


} ///:~ 





这 个 例子 回 我 们 证 明了 使 用 private 的 方便 : 有 时 可 能 想 控制 对 象 的 创建 
方式 ， 并 防止 有 人 直接 访问 一 个 特定 的 构建 器 (或 者 所 有 构建 器 ) 。 在 
上 面 的 例子 中 ， 我 们 不 可 通过 它 的 构建 器 创建 一 个 Sundae 对 象 ， 相反 ， 
必须 调用 makeASundae() 方 法 来 实现 (注释 @))。 


O: 此 时 还 会 产生 另 一 个 影响 : 由 于 默认 构建 器 是 唯一 获得 定义 的 ， 而 
日 它 的 属性 是 private， 所 以 可 防止 对 这 个 类 的 继承 (这 是 第 6 章 要 重点 
讲述 的 主题 ) 。 


各 确定 一 个 类 只 有 一 个 “助手 ?方法 ， 那 么 对 于 任何 方法 来 说 ， 都 可 以 把 
它们 设 为 private， 从 而 保证 上 自己 不 会 误 在 包 内 其 他 地 方 使 用 它 ， 防 止 目 
己 更 改 或 删除 方法 。 将 一 个 方法 的 属性 设 为 private 后 ， 可 保证 目 己 一 直 
保持 这 一 选项 (然而 ， 右 一 个 句柄 被 设 为 private， 并 不 表明 其 他 对 象 不 
4 有 指 同 同一 个 对 象 的 public 句 柄 。 有 天“ 别名 ”的 问题 将 在 第 12 章 详 
IW) . 


5.2.4 protected: “友好 的 一 种 ” 


protected〈 受 到 保护 的 ) 访问 指示 符 要 求 大 家 提前 有 所 认识 。 首 先 应 注 
意 这 样 一 个 事实 : 为 继续 学 习 本 书 一 直到 继承 那 一 章 之 前 的 内 容 ， 并 不 
一 定 需要 先 理解 本 小 节 的 内 容 。 但 为 了 保持 内 容 的 完整 ， 这 儿 仍 然 要 对 
此 进行 简要 说 明 ， 并 提供 相关 的 例子 。 


protected 关 键 字 为 我 们 引入 了 一 种 名 为 “继承 ”的 概念 ， 它 以 现 有 的 类 为 
基础 ， 并 在 其 中 加 入 新 的 成 员 ， 同 时 不 会 对 现 有 的 类 产生 影响 我 们 
将 这 种 现 有 的 类 称 为 “基础 类 ”或 者 “基本 类 ”(Base Class) 。 亦 可 改变 那 
个 类 现 有 成 员 的 行为 。 对 于 从 一 个 现 有 类 的 继承 ， 我 们 说 自己 的 新 

类 “扩展 ”(extends ) 了 那个 现 有 的 类 。 如 下 所 示 : 


























class Foo extends Bar { 
类 定义 剩余 的 部 分 看 起 来 是 完全 相同 的 。 


若 新 建 一 个 包 ， 并 从 男 一 个 包 内 的 某 个 类 里 继承 ， 则 唯一 能 够 访问 的 成 
员 就 是 原来 那个 包 的 public 成 员 。 当 然 ， 如 果 在 相同 的 包 里 进行 继承 ， 

那么 继承 获得 的 包 能 够 访问 所 有 “友好 ”的 成 员 。 有 些 时 候 ， 基 础 类 的 创 
建 者 喜欢 提供 一 个 特殊 的 成 员 ， 并 人 允许 访问 衍生 类 。 这 正 是 protected 的 
工作 。 若 往 回 引用 5.2.2 小 节 “public: 接口 访问 ”的 那个 Cookie.java 文 件 ， 
则 下 面 这 个 类 就 不 能 访问 “友好 ”的 成 员 : 


//: ChocolateChip.java 





// Can't access friendly member 
// in another class 
import c05.dessert.*; 
public class ChocolateChip extends Cookie { 
public ChocolateChip() { 
System.out.printin( 
"ChocolateChip constructor"); 
} 
public static void main(String[] args) { 
ChocolateChip x = new ChocolateChip(); 
//! x.f00(); // Can't access foo 


} 
7 二 


对 于 继承 ， 值 得 注意 的 一 件 有 趣 的 事情 是 倘若 方法 foo0 存 在 于 类 Cookie 


中 ， 那 么 它 也 会 存在 于 从 Cookie 继 承 的 所 有 类 中。 但 由 于 foo0 在 外 部 的 
包 里 是 “友好 ”的 ， 所 以 我 们 不 能 使 用 它 。 当 然 ， 亦 可 将 其 变 成 public。 
但 这 样 一 来 ， 由 于 所 有 人 都 能 自由 访问 它 ， 所 以 可 能 并 非 我 们 所 希望 的 
局 面 。 知 象 下 面 这 样 修改 类 Cookie: 


public class Cookie { 


public Cookie() { 


System.out.printin( "Cookie constructor"); 
} 
protected void foo() { 


System.out.println("foo"); 


那么 仍然 能 在 包 dessert 里 “友好 ”地 访问 foo()， 但 从 Cookie 继 承 的 其 他 东 
西 亦 可 自由 地 访问 它 。 然 而 ， 它 并 非 公 共 的 (public)。 


5.3 接口 与 实现 


我 们 通常 认为 访问 控制 是 “隐藏 实施 细节 ”的 一 种 方式 。 将 数据 和 方法 封 
装 到 类 内 后 ， 可 生成 一 种 数据 类 型 ， 它 具有 上 自己 的 特征 与 行为 。 但 由 于 
两 方面 重要 的 原因 ， 访 问 为 那个 数据 类 型 加 上 了 自己 的 边界 。 第 一 个 原 
因 是 规定 客户 程序 员 哪 些 能 够 使 用 ， 哪 些 不 能 。 我 们 可 在 结构 里 构建 日 
己 的 内 部 机 制 ， 不 用 担心 客户 程序 员 将 其 当 作 接口 的 一 部 分 ， 从 而 自由 
地 使 用 或 者 “滥用 ”。 


这 个 原因 直接 导致 了 第 二 个 原因 : 我 们 需要 将 接口 同 实施 细节 分 离开 。 
在 结构 在 一 系列 程序 中 使 用 ， 但 用 户 除 了 将 消息 发 给 public 接 口 之 外 ， 
不 能 做 其 他 任何 事情 ， 我 们 束 可 以 改变 不 属于 public 的 所 有 东西 (如 “ 友 
的 ”、Pprotected 以 及 private) ， 同 时 不 要 求 用 户 对 他 们 的 代码 作 任何 修 
Mo 














我 们 现在 是 在 一 个 面 癌 对 象 的 编程 环境 中 ， 其 中 的 一 个 类 (class〉 实际 
古 指 “ 一 类 对 象 ?”， 就 象 我 们 说 “ 鱼 类 ”或 “ 乌 类 ”那样 。 从 属于 这 个 类 的 所 
有 对 象 部 共享 这 些 特征 与 行为 。“ 类 ”是 对 属于 这 一 类 的 所 有 对 象 的 外 观 
及 行为 进行 的 一 种 描述 。 


在 一 些 早期 DOP 语言 中 ， 如 Simula-67， 关 键 字 class 的 作用 是 描述 一 种 新 
的 数据 类 型 。 同 样 的 关键 字 在 大 多 数 面 同 对 象 的 编程 语言 里 都 得 到 了 应 
用 。 它 其 实 是 整个 语言 的 焦点 : 需要 新 建 数 据 类 型 的 场合 比 那 些 用 于 容 
纳 数 据 和 方法 的 “容器 ”多 得 多 。 


在 Java 中 ， 类 是 最 基本 的 OOP 概 念 。 它 是 本 书 未 采用 粗 体 印刷 的 关键 字 
之 一 一 一 由 于 数量 太 多 ， 所 以 会 造成 页 面 排版 的 严重 混乱 。 


为 清楚 起 见 ， 可 考虑 用 特殊 的 样式 创建 一 个 类 : 将 public 成 员 置 于 最 开 
头 ， 后 面 跟随 protected、 友 好 以 及 private 成 员 。 这 样 做 的 好 处 是 类 的 使 
用 者 可 从 上 辣 下 依次 阅读 ， 并 首先 看 到 对 自己 来 说 最 重要 的 内 容 ( 即 
public 成 员 ， 因 为 它们 可 从 文件 的 外 部 访问 ) ， 并 在 直到 非 公 共 成 员 后 
停止 疯 读 ， 后 者 已 经 属于 内 部 实施 细节 的 一 部 分 了 。 然 而 ， 利 用 由 
javadoc 提 供 支 持 的 注释 文档 (已 在 第 2 章 介 绍 ) ， 代 码 的 可 读 性 问题 已 
在 很 大 程度 上 得 到 了 解决 。 


public class X { 


























public void pub1( ) { /* . . . */ } 


public void pub2( ) { /* . .. */ } 
public void pub3( ) { /* . .. */ } 
private void privi( ) { /* . . . */ } 
private void priv2( ) { /* . . . */ } 
private void priv3( ) { /* .. . */ } 


private int 1; 


LI aoa 


由 于 接口 和 实施 细节 仍然 混合 在 一 起 ， 所 以 只 是 部 分 容易 阅读 。 也 就 是 
说 ， 仍 然 能 够 看 到 源码 一 一 实施 的 细节 ， 因 为 它们 需要 保存 在 类 里 面 。 
问 一 个 类 的 消费 者 显示 出 接口 实际 是 “类 浏览 器 ”的 工作 。 这 种 工具 能 奉 
找 所 有 可 用 的 类 ， 总 结 出 可 对 它们 采取 的 全 部 操作 《比如 可 以 使 用 哪些 

员 等 ) ， 并 用 一 种 清爽 悦目 的 形式 显示 出 来 。 到 大 家 该 到 这 本 书 的 时 
候 ， 所 有 优秀 的 Java 开 发 工具 都 应 推出 了 自己 的 浏览 器 。 
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5.4 类 访问 


在 Java 中 ， 亦 可 用 访问 指示 符 判 断 出 一 个 库 内 的 哪些 类 可 由 那个 库 的 用 
户 使 用 。 知 想 一 个 类 能 由 客户 程序 员 调 用 ， 可 在 类 主体 的 起 始 花 括号 前 
面 某 处 放置 一 个 public 关 键 字 。 它 控制 着 客户 程序 员 是 否 能 够 创建 属于 
这 个 类 的 一 个 对 象 。 

为 控制 一 个 类 的 访问 ， 指 示 符 必须 在 关键 字 class 之 前 出 现 。 所 以 我 们 能 
够 使 用 : 


public class Widget { 
也 就 是 说 ， 假 若 我 们 的 库 名 是 mylib， 那 么 所 有 客户 程序 员 都 能 访问 

















Widget 一 一 通过 下 述 语句 : 
import mylib.Widget; 
或 者 


import mylib.*; 
然而 ， 我 们 同时 还 要 注意 到 一 些 额外 的 限制 : 


(1) 每 个 编译 早 元 (文件 ) 都 只 能 有 一 个 public 类 。 每 个 编译 单元 有 一 个 
公共 接口 的 概念 是 由 那个 公共 类 表达 出 来 的 。 根 据 自己 的 需要 ， 它 可 拥 
有 任意 多 个 提供 文 撑 的 “友好 ?类 。 但 知 在 一 个 编译 单元 里 使 用 了 多 个 
public 类 ， 编 译 需 束 会 问 我 们 提示 一 条 出 错 消息 。 


(2) public 类 的 名 字 必 须 与 包含 了 编译 单元 的 那个 文件 的 名 字 完 全 相符 ， 
甚至 包括 它 的 大 小 写 形 式 。 所 以 对 于 Widget 来 说 ， 文 件 的 名 字 必 须 是 
Widget.java， 而 不 应 是 widget,java 或 者 WIDGET.java。 同 样 地 ， 如 果 出 
现 不 符 ， 束 会 报告 一 个 编译 期 错误 。 


(3) ARE CSF WL) 有 一 个 编译 单元 根本 没有 任何 公共 类 。 此 时 ， 可 
按 上 自己 的 意愿 任意 指定 文件 名 。 


如 果 已 经 获得 了 mylib 内 部 的 一 个 类 ， 准 备用 它 完成 由 Widget 或 者 mylib 




















内 部 的 其 他 某 些 public 类 执行 的 任务 ， 此 时 又 会 出 现 什 么 情况 呢 ? 我 们 
不 布 望花 费力 气 为 客户 程序 员 编 制 文档 ， 并 感觉 以 后 人 条 个 时 候 也 许 会 进 
行 大 手笔 的 修改 ， 并 将 自己 的 类 一 起 删 控 ， 换 成 为 一 个 不 同 的 类 。 为 获 
得 这 种 灵活 处 理 的 能 力 ， 需 要 保证 没有 客户 程序 员 能 够 依赖 目 己 隐藏 于 
mylib 内 部 的 特定 实施 细节 。 为 达到 这 个 目的 ， 只 需 将 public 关 键 字 从 关 
中 剔除 即 可 ， 这 样 便 把 类 变 成 了 “友好 的 ”《〈 类 仅 能 在 包 内 使 用 ) 。 


注意 不 可 将 类 设 成 private〈 那 样 会 使 除 类 之 外 的 其 他 东西 都 不 能 访问 
它 ) ， 也 不 能 设 成 protected GERO) 。 因 此 ， 我 们 现在 对 于 类 的 访问 
只 有 两 个 选择 :“ 友 好 的 ”或 者 public。 若 不 愿 其 他 任何 人 访问 那个 类 ， 
可 将 所 有 构建 器 设 为 private。 这 样 一 来 ， 在 类 的 一 个 static 成 员 内 部 ， 除 
QRO) 。 
0 下 例 所 示 : 


//: Lunch.java 





// Demonstrates class access specifiers. 
// Make a class effectively private 
// with private constructors: 
class Soup { 

private Soup() {} 

// (1) Allow creation via static method: 
public static Soup makeSoup() { 

return new Soup(); 

} 

// (2) Create a static object and 
// return a reference upon request. 
// (The "Singleton" pattern): 


private static Soup psi = new Soup(); 


public static Soup access() { 
return ps1; 

} 

public void f() {} 
} 
class Sandwich { // Uses Lunch 
void f() { new Lunch(); } 
} 
// Only one public class allowed per file: 
public class Lunch { 

void test() { 


// Can't do this! Private constructor: 


//! Soup privi = new Soup(); 
Soup priv2 = Soup.makeSoup(); 
Sandwich f1 = new Sandwich(); 
Soup.access().f(); 


} 
} ///:~ 


©: 实际 上 ，Java ”1.1 内 部 类 既 可 以 是 “受到 保护 的 "， 也 可 以 是 “私有 
的 "， 但 那 属于 特别 情况 。 第 7 章 会 详细 解释 这 个 问题 。 
©: 亦 可 通过 从 那个 类 继承 来 实现 。 


迄今 为 上 上， 我 们 创建 过 的 大 多 数 方法 都 是 要 么 返回 void， 要 么 返回 一 1 
基本 数据 类 型 。 所 以 对 下 述 定义 来 说 : 


public static Soup access() { 
return psl; 
} 


它 最 开始 多 少 会 使 人 有 些 迷 惑 。 位 于 方法 名 〈access) 前 的 单词 指出 方 
法 到 底 返 回 什么 。 在 这 之 前 ， 我 们 看 到 的 都 是 void， 它 意味 着 “什么 也 
不 返回 ”(void 在 英语 里 是 “虚无 2 的 意思 。 但 亦 可 返回 指 同 一 个 对 象 的 句 
T ia 该 方法 返回 一 个 句柄 ， 它 指 同类 Soup 的 
一 个 对 家 。 


Soup 类 辣 我 们 展示 出 如 何 通 过 将 所 有 构建 器 都 设 为 private， 从 而 防止 直 
接 创 建 一 个 类 。 请 记 住 ， 假 知 不 明确 地 至 少 创建 一 个 构建 问 ， 束 会 目 动 
创建 默认 构建 器 (没有 自 变 量 ) 。 大 自己 编写 默认 构建 右 ， 它 束 不 会 自 
动 创 建 。 把 它 变 成 private 后 ， 就 没 人 能 为 那个 类 创建 一 个 对 象 。 但 别人 
怎样 使 用 这 个 类 呢 ? 上 面 的 例子 为 我 们 揭示 出 了 两 个 选择 。 
择 ， 我 们 可 创建 一 个 static 方 法 ， 再 通过 它 创 建 一 个 新 的 Soup， 然 后 返 

指 同 它 的 一 个 句柄 。 如 果 想 在 返回 之 前 对 Soup 进 行 一 些 额 外 的 操作 ， 
者 想 了 解 准 备 创建 多 少 个 Soup 对 象 ( 可 能 是 为 了 限制 它们 的 个 数 ) ， 这 
种 方案 无 疑 是 特别 有 用 的 。 


第 二 个 选择 是 采用 “设计 方案 ”(Design Pattern) 技术 ， 本 书后 面 会 对 此 
进行 详细 介绍 。 通 常 方案 叫 作 : ee 因为 它 仅 允许 创建 一 个 对 象 。 类 
Soup BO Soup ‘static privates in, PT 以 有 一 个 而 且 只 

能 有 一 个 。 除 非 通过 public 方 法 access0， 和 否则 根本 无 法 访问 它 。 


正如 早先 指出 的 那样 ， 如 果 不 针 对 类 的 访问 设置 一 个 访问 指示 符 ， 那 么 
它 会 自动 默认 为 “友好 的 "。 这 意味 着 那个 类 的 对 象 可 由 包 内 的 其 他 类 创 
建 ， 但 不 能 由 包 外 创建 。 请 记 住 ， 对 于 相同 目录 内 的 所 有 文件 ， 如 果 没 
有 明确 地 进行 package 声 明 ， 那 么 它们 都 默认 为 那个 目录 的 默认 包 的 一 
部 分 。 然 而 ， 假 若 那 个 类 一 个 static 成 员 的 属性 是 public， 那 么 客户 程序 
a reece 即使 它们 不 能 创建 属于 那个 类 的 一 个 
$ 








5.5 总 结 


对 于 任何 关系 ， 最 重要 的 一 点 都 是 规定 好 所 有 方面 都 必须 遵守 的 界限 或 
规则 。 创 建 一 个 库 时 ， 相 当 于 建立 了 同 那个 库 的 用 户 《“ 即 “客户 程序 
R 的 一 种 关系 一 一 那些 用 户 属 于 另外 的 程序 员 ， 可 能 用 我 们 的 库 自 
行 构建 一 个 应 用 程序 ， 或 者 用 我 们 的 库 构 建 一 个 更 大 的 库 。 


如 果 不 制订 规则 ， 客 户 程序 员 就 可 以 随心 所 和 欲 地 操作 一 个 类 的 所 有 成 
员 ， 无 论 我 们 本 来 愿 不 愿意 其 中 的 一 些 成 员 被 直接 操作 。 所 有 东西 部 在 
别人 面前 都 暴露 无 遗 。 


本 章 讲述 了 如 何 构 建 类 ， 从 而 制作 出 理想 的 库 。 首 先 ， 我 们 讲述 如 何 将 
一 组 类 封装 到 一 个 库 里 。 其 次 ， 我 们 讲述 类 如 何 控制 对 目 己 成 员 的 访 
问 。 


一 般 情 况 下 ， 一 个 C 程 序 项 目 会 在 50K 到 100K 行 代码 之 间 的 某 个 地 方 开 
始 中 断 。 这 是 由 于 C 仅 有 一 个 “命名 空间 ?， 所 以 名 字 会 开始 互相 抵触 ， 

从 而 造成 额外 的 管理 开销 。 而 在 Java 中 ，package 关 键 字 、 包 命名 方案 以 
及 import 关 键 字 为 我 们 提供 对 名 字 的 完全 控制 ， 所 以 命名 冲突 的 问题 可 
以 很 轻易 地 得 到 避免 。 


有 两 方面 的 原因 要 求 我 们 控制 对 成 员 的 访问 。 第 一 个 是 防止 用 户 接触 那 
些 他 们 不 应 磁 的 工具 。 对 于 数据 类 型 的 内 部 机 制 ， 那 些 工具 是 必需 的 。 
但 它们 并 不 属于 用 户 接口 的 一 部 分 ， 用 户 不 必用 它 来 解决 目 己 的 特定 问 
题 。 所 以 将 方法 和 字段 变 成 < 私有 ” (private) 后 ， 可 极 大 方便 用 户 。 因 
为 他 们 能 轻易 看 出 哪些 对 于 上 自己 来 说 是 最 重要 的 ， 以 及 哪些 是 目 己 需要 
忽略 的 。 这 样 便 简 化 了 用 户 对 一 个 类 的 理解 。 


进行 访问 控制 的 第 二 个 、 也 是 最 重要 的 一 个 原因 是 : 允许 库 设 计 者 改变 
类 的 内 部 工作 机 制 ， 同 时 不 必 担 心 它 会 对 客户 程序 员 产 生 什么 影响 。 最 
开始 的 时 候 ， 可 用 一 种 方法 构建 一 个 类 ， 后 来 发 现 需要 重新 构建 代码 ， 

以 便 达 到 更 快 的 速度 。 如 接口 和 实施 细节 早已 进行 了 明确 的 分 隔 与 保 

护 ， 就 可 以 轻松 地 达到 目 己 的 目的 ， 不 要 求 用 户 改 写 他 们 的 代码 。 


利用 Java 中 的 访问 指示 符 ， 可 有 效 控制 类 的 创建 者 。 那 个 类 的 用 户 可 确 
切 知道 哪些 是 自己 能 够 使 用 的 ， 哪 些 则 是 可 以 忽略 的 。 但 更 重要 的 一 点 















































是 ， 它 可 确保 没有 任何 用 户 能 依赖 一 个 类 的 基础 实施 机 制 的 任何 部 分 。 
作为 一 个 类 的 创建 者 ， 我 们 可 目 由 修改 基础 的 实施 细节 ， 这 一 改变 不 会 
对 客户 程序 员 产 生 任 何 影响 ， 因 为 他 们 不 能 访问 类 的 那 一 部 分 。 


有 能 力 改变 基础 的 实施 细节 后 ， 除 了 能 在 以 后 改进 自己 的 设置 之 外 ， 也 
同时 拥有 了 “犯错 误 ” 的 自由 。 无 论 当 初 计 划 与 设计 时 有 多 么 仔细 ， 仍 然 
有 可 能 出 现 一 些 失 误 。 由 于 知道 自己 能 相当 安全 地 犯 下 这 种 错误 ， 所 以 
可 以 放心 大 胆 地 进行 更 多 、 更 自由 的 试验 。 这 对 自己 编程 水 平 的 提高 是 
很 有 帮助 的 ， 使 整个 项 目 最 终 能 更 快 、 更 好 地 完成 。 


一 个 类 的 公共 接口 是 所 有 用 户 都 能 看 见 的 ， 所 以 在 进行 分 析 与 设计 的 时 
候 ， 这 是 应 尽量 保证 其 准确 性 的 最 重要 的 一 个 部 分 。 但 也 不 必 过 于 紧 
张 ， 少 许 的 误差 仍然 是 允许 的 。 硝 最 初 设计 的 接口 存在 少许 问题 ， 可 考 
etre ree ie erg eee 
k 和 东西。 























5.6 练习 


(1) 用 public、Pprivate、Pprotected 以 及 “友好 的 ”数据 成 员 及 方法 成 员 创建 
一 个 类 。 创 建 属于 这 个 类 的 一 个 对 象 ， 并 观察 在 试图 访问 所 有 类 成 员 时 
会 获得 哪 种 类 型 的 编译 器 错误 提示 。 注 意 同一 个 目录 内 的 类 属于 “ 默 

认 ” 包 的 一 部 分 。 

(2) 用 protected 数 据 创建 一 个 类 。 在 相同 的 文件 里 创建 第 二 个 类 ， 用 一 个 
方法 操纵 第 一 个 类 里 的 protected 数 据 。 


(3) 新 建 一 个 目录 ， 并 编辑 自己 的 CLASSPATH， 以 便 包 括 那 个 新 目录 。 
将 P.class 文 件 复制 到 自己 的 新 目录 ， 然 后 改变 文件 名 、P 类 以 及 方法 名 
( 亦 可 考虑 添加 额外 的 输出 ， 观 察 它 的 运行 过 程 ) 。 在 一 个 不 同 的 目录 
里 创建 另 一 个 程序 ， 令 其 使 用 自己 的 新 类 。 
(4) 在 c05 目 录 (假定 在 自己 的 CLASSPATH 里 ) 创建 下 述 文件 : 
214 页 程序 
然后 在 c05 之 外 的 另 一 个 目录 里 创建 下 述 文件 : 
214-215 页 程序 


解释 编译 器 为 什么 会 产生 一 个 错误 。 将 Foreign《〈 外 部 ) 类 作为 c05 包 的 
一 部 分 改变 了 什么 东西 吗 ? 














第 6 章 类 再生 


“Java 引 人 注目 的 一 项 特性 是 代码 的 重复 使 用 或 者 再 生 。 但 最 具 章 合意 
义 的 是 ， 除 代码 的 复制 和 修改 以 外 ， 我 们 还 能 做 多 得 多 的 其 他 事情 。” 


在 象 C 那 样 的 程序 化 语言 里 ， 代 码 的 重复 使 用 早已 可 行 ， 但 效果 不 是 特 
别 显 车 。 与 Java 的 其 他 地 方 一 样 ， 这 个 方案 解决 的 也 是 与 类 有 关 的 问 
题 。 我 们 通过 创建 新 类 来 重复 使 用 代码 ， 但 却 用 不 痢 重 新 创建 ， 可 以 直 
接 使 用 别人 已 建 好 并 调试 好 的 现成 类 。 


但 这 样 做 必须 保证 不 会 干扰 原 有 的 代码 。 在 这 一 章 里 ， 我 们 将 介绍 两 个 
达到 这 一 目标 的 方法 。 第 一 个 最 简单 : 在 新 类 里 简单 地 创建 原 有 类 的 对 
象 。 我 们 把 这 种 方法 叫 作 ”合成 ? 因为 新 类 由 现 有 类 的 对 象 合并 而 成 。 
我 们 只 是 简单 地 重复 利用 代码 的 功能 ， 而 不 是 采用 它 的 形式 。 


第 二 种 方法 则 显得 稍微 有 些 技巧 。 它 创建 一 个 新 类 ， 将 其 作为 现 有 类 的 
一 个 “类 型 >”。 我 们 可 以 原样 采取 现 有 类 的 形式 ， 并 在 其 中 加 入 新 代码 ， 
同时 不 会 对 现 有 的 类 产生 影响 。 这 种 魔术 般 的 行为 叫 作 “ 继 

7x” (Inheritance) ， 涉 及 的 大 多 数 工 作 都 是 由 编译 器 完成 的 。 对 于 面 问 
对 象 的 程序 设计 , “继承 ?是 最 重要 的 基础 概念 之 一 。 它 对 我 们 下 一 章 要 
讲述 的 内 容 会 产生 一 些 额外 的 影响 。 

对 于 合成 与 继承 这 两 种 方法 ， 大 多 数 语法 和 行为 都 是 类 似 的 《因为 它们 
都 要 根据 现 有 的 类 型 后 成 新 类 型 ) 。 在 本 章 ， 我 们 将 深入 学 习 这 些 代码 
再 生 或 者 重复 使 用 的 机 制 。 

















6.1 合成 的 语法 


就 以 前 的 学 习 情 况 来 看 ， 事 实 上 已 进行 了 多 次 “合成 ?操作 。 为 进行 合 
成 ， 我 们 只 需 在 新 类 里 简单 地 置 入 对 象 句 柄 即 可 。 举 个 例子 来 说 ， 假 定 
需要 在 一 个 对 象 里 容纳 儿 个 String 对 象 、 两 种 基本 数据 类 型 以 及 属于 为 
一 个 类 的 一 个 对 象 。 对 于 非 基 本 类 型 的 对 象 来 说 ， 只 需 将 句柄 置 于 新 类 
即 可 ; 而 对 于 基本 数据 类 型 来 说 ， 则 需 在 自己 的 类 中 定义 它们 。 如 下 所 
示 〔 寿 执行 该 程序 时 有 及 烦 ， 请 参见 第 3 章 3.1.2 小 市 “赋值 ”) : 


//: SprinklerSystem. java 

















// Composition for code reuse 
package c06; 
class WaterSource { 
private String s; 
WaterSource() { 
System.out.println("WaterSource()"); 
s = new String("Constructed"); 
} 
public String toString() { return s; } 
} 
public class SprinklerSystem { 
private String valve1, valve2, valve3, valve4; 
WaterSource source; 
int 1; 


float f; 


void print() { 


System.out.printin("valvet " + valve); 


System.out.printin("valve2 " + Valve2 ) ; 


System.out.printin("valve3 " + valve3); 


l 
+ 


System.out.println("valve4 = " valve4); 
System.out.printlin("i = " + i); 
System.out.printin("f = " + f); 
System.out.println("source = " + source); 

} 

public static void main(String[] args) { 
SprinklerSystem x = new SprinklerSystem(); 
X.print(); 

} 


} ///:~ 


WaterSource 内 定义 的 一 个 方法 是 比较 特别 的 : toString()。 大 家 不 久 就 会 
知道 ， 每 种 非 基 本 类 型 的 对 象 都 有 一 个 toString() 方 法 。 寿 编译 器 本 来 希 
望 一 个 String， 但 却 获 得 某 个 这 样 的 对 象 ， 就 会 调用 这 个 方法 。 所 以 在 
下 面 这 个 表达 式 中 : 








System.out.println("Source = " + source) ; 


编译 器 会 发 现 我 们 试图 网 一 个 WaterSource 添 加 一 个 String 对 象 〈"source 
=") 。 这 对 它 来 说 是 不 可 接受 的 ， 因 为 我 们 只 能 将 一 个 字 串 “添加 ”到 另 
一 个 字 串 ， 所 以 它 会 说 : “我 要 调用 toString0， 把 source 转 换 成 字 

串 ! ”经 这 样 处 理 后 ， 它 就 能 编译 两 个 字 串 ， 并 将 结果 字 串 传递 给 一 个 
System.out.printIn()。 每 次 随同 自己 创建 的 一 个 类 允许 这 种 行为 的 时 候 ， 
都 只 需要 写 一 个 toString0 方 法 。 








如 果 不 深究 ， 可 能 会 草率 地 认为 编译 器 会 为 上 述 代 码 中 的 每 个 句柄 都 自 
动 构造 对 象 ( 由 于 Java 的 安全 和 谨慎 的 形象 )。 例 如 ， 可 能 以 为 它 会 为 
WaterSource 调 用 默认 构建 占 ， 以 便 初 始 化 source。 打 印 语句 的 输出 事实 


上 是 : 


valvei = null 


valve2 = null 
valve3 = null 


valve4 = null 


source = null 


TERA VENA BODINE AR oe SO, AR Ee Ta HY AB 
样 。 但 对 象 句柄 会 初始 化 成 nul。 而 且 假 耕 试 图 为 它们 中 的 任何 一 个 调 
用 方法 ， 就 会 产生 一 次 “违例 ”。 这 种 结果 实际 是 相当 好 的 《而 且 很 有 
用 ) ， 我 们 可 在 不 丢弃 一 次 违例 的 前 提 下 ， 仍 然 把 它们 打印 出 来 。 
编译 器 并 不 只 是 为 每 个 句柄 创建 一 个 默认 对 象 ， 因 为 那样 会 在 许多 情况 
下 招致 不 必要 的 开销 。 如 希望 句柄 得 到 初始 化 ， 可 在 下 面 这 些 地 方 进 
位: 


(1) FEXT ATE SATIN AG © XERE E TEE ed E a Dd FZ BE FS BI) 
始 化 o 











(2) 在 那个 类 的 构建 器 中 。 


(3) ” 紧 徘 在 要 求实 际 使 用 那个 对 象 之 前 。 这 样 做 可 减少 不 必要 的 开销 
一 一 假如 对 象 并 不 需要 创建 的 话 。 


下 面向 大 家 展示 了 所 有 这 三 种 方法 : 


//: Bath.java 








// Constructor initialization with composition 
class Soap { 
private String s; 
Soap() { 
System.out.println("Soap()"); 
s = new String("Constructed"); 


} 


public String toString() { return s; } 
} 
public class Bath { 

private String 


// Initializing at point of definition: 


si = new String("Happy"), 
s2 = "Happy", 
s3, S4; 
Soap castille; 
int i; 
float toy; 
Bath() { 
System.out.println("Inside Bath()"); 
s3 = new String("Joy"); 


i = 47; 


toy = 3.14f; 
castille = new Soap(); 
} 
void print() { 
// Delayed initialization: 
if(s4 == null) 


s4 = new String("Joy"); 


System.out.printin("si = " + s1); 
System.out.println("s2 = " + s2); 
System.out.printin("s3 = " + s3); 
System.out.printin("s4 = " + s4); 
System.out.printlin("i = " + i); 
System.out.println("toy = " + toy); 
System.out.println("castille = " + castille); 


} 
public static void main(String[] args) { 
Bath b = new Bath(); 
b.print(); 
} 
LIL 


请 注意 在 Bath 构 建 咒 中， 在 所 有 初始 化 开始 之 前 执行 了 一 个 语句 。 如 果 
不 在 定义 时 进行 初始 化 ， 仍 然 不 能 保证 能 在 将 一 条 消息 用 给 一 个 对 象 名 
柄 之 前 会 执行 任何 初始 化 一 一 除非 出 现 不 可 避免 的 运行 期 违例 。 








下 面 是 该 程序 的 输出 : 


Inside Bath() 


Soap() 

si = Happy 
s2 = Happy 
s3 = Joy 
s4 = Joy 

i = 47 

toy = 3.14 


castille = Constructed 


aa tee ped a ne 


6.2 继承 的 语法 


继承 与 Java〈 以 及 其 他 OOP 语 言 ) 非 营 紧密 地 结合 在 一 起 。 我 们 早 在 第 
1 章 就 为 大 家 引入 了 继承 的 概念 ， 并 在 那 章 之 后 到 本 章 之 前 的 各 章 里 不 
时 用 到 ， 因 为 一 些 特殊 的 场合 要 求 必须 使 用 继承 。 除 此 以 外 ， 创 建 一 个 
类 时 肯定 会 进行 继承 ， 因 为 吾 非 如 此 ， 会 从 Java 的 标准 根 类 Object 中 继 
不 。 


用 于 合成 的 语法 是 非常 简单 且 直 观 的 。 但 为 了 进行 继承 ， 必 须 采 用 一 种 
全 然 不 同 的 形式 。 需 要 继承 的 时 候 ， 我 们 会 说 :“ 这 个 新 类 和 那个 旧 类 
差不多 。” 为 了 在 代码 里 表面 这 一 观念 ， 需 要 给 出 类 名 。 但 在 类 主体 的 
起 始 花 括 号 之 前 ， 需 要 放置 一 个 关键 字 extends， 在 后 面 跟随 “基础 类 ”的 
名 字 。 知 采取 这 种 做 法 ， 就 可 自动 获得 基础 类 的 所 有 数据 成 员 以 及 方 
ae 下 面 是 一 个 例子 : 


//: Detergent.java 

















// Inheritance syntax & properties 

class Cleanser { 
private String s = new String("Cleanser"); 
public void append(String a) { s += a; } 
public void dilute() { append(" dilute()"); } 
public void apply() { append(" apply()"); } 
public void scrub() { append(" scrub()"); } 
public void print() { System.out.printin(s); } 
public static void main(String[] args) { 

Cleanser x = new Cleanser(); 


x.dilute(); x.apply(); x.scrub(); 


X.print(); 
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public class Detergent extends Cleanser { 
// Change a method: 
public void scrub() { 
append(" Detergent.scrub()"); 
super.scrub(); // Call base-class version 
} 
// Add methods to the interface: 
public void foam() { append(" foam()"); } 
// Test the new class: 
public static void main(String[] args) { 
Detergent x = new Detergent(); 
x.dilute(); 
x.apply(); 
x.scrub(); 
x.foam(); 
X.print(); 
System.out.printin("Testing base class:"); 


Cleanser .main(args); 


de 





这 个 例子 向 大 家 展示 了 大 量 特性 。 首 先 ， 在 Cleanser 。 append() 方 法 里 ， 
字 串 同一 个 s 连 接 起 来 。 这 是 用 “+=" 运 算 符 实现 的 。 同 “+” 一 样 ，“+=" 被 
Java 用 于 对 字 串 进行 “过 载 ”处 理 。 


其 次 ， 无 论 Cleanser 还 是 Detergent 都 包含 了 一 个 main(0) 方 法 。 我 们 可 为 目 
己 的 每 个 类 都 创建 一 个 main()。 通 党 建议 大 家 象 这 样 进行 编写 代码 ， 使 
自己 的 测试 代码 能 够 封装 到 类 内 。 即 便 在 程序 中 含有 数量 众多 的 类 ， 但 
对 于 在 命令 行 请 求 的 public 类 ， 只 有 main0 才 会 得 到 调用 。 所 以 在 这 种 情 
况 下 ， 当 我 们 使 用 *java ”Detergent” 的 时 候 ， 调 用 的 是 Degergent.main() 
一 一 即使 Cleanser 并 非 一 个 public 类 。 采 用 这 种 将 main0 置 入 每 个 类 的 做 
法 ， 可 方便 地 为 每 个 类 都 进行 单元 测试 。 而 且 在 完成 测试 以 后 ， 毋 需 将 
main0 删 去 ， 可 把 它 保 留 下 来 ， 用 于 以 后 的 测试 。 


在 这 里 ， 大 家 可 看 到 Deteregent.main() 对 Cleanser.main() 的 调用 是 明确 进 
行 的 。 


需要 痢 重 强调 的 是 Cleanser 中 的 所 有 类 都 是 public 属 性 。 请 记 住 ， 倘 行 省 
略 所 有 访问 指示 符 ， 则 成 员 默 认为 “友好 的 "。 这 样 一 来 ， 就 只 允许 对 包 
成 员 进 行 访问 。 在 这 个 包 内 ， 任 何人 都 可 使 用 那些 没有 访问 指示 符 的 方 
法 。 例 如 ，Detergent 将 不 会 遇 到 任何 矿 烦 。 然 而 ， 假 设 来 和 目 另外 某 个 包 
的 类 准备 继承 Cleanser， 它 就 只 能 访问 那些 public 成 员 。 所 以 在 计划 继承 
的 时 候 ， 一 个 比较 好 的 规则 是 将 所 有 字段 都 设 为 private， 并 将 所 有 方法 
都 设 为 public 〈protected 成 员 也 人 允许 衍生 出 来 的 类 访问 它 ; 以 后 还 会 深 

入 探讨 这 一 问题 ) 。 当 然 ， 在 一 些 特殊 的 场合 ， 我 们 仍然 必须 作出 一 些 
调整 ， 但 这 并 不 是 一 个 好 的 做 法 。 


注意 Cleanser 在 它 的 接口 中 含有 一 系列 方法 : append0，dilute0)， 
applyO0，scrub0O 以 及 print0。 由 于 Detergent 是 从 Cleanser 衍 生出 来 的 〈 通 
过 extends 关 键 字 ) ， 上 所 以 它 会 自动 获得 接口 内 的 所 有 这 些 方法 即使 
我 们 在 Detergent 里 并 未 看 到 对 它们 的 明确 定义 。 这 样 一 来 ， 融 可 将 继承 
想象 成 “对 接口 的 重复 利用 ”或 者 “接口 的 再 生 ”( 以 后 的 实施 细节 可 以 有 自 
由 设置 ， 但 那 并 非 我 们 强调 的 重点 〉。 


正如 在 scrubO 里 看 到 的 那样 ， 可 以 获得 在 基础 类 里 定义 的 一 个 方法 ， 并 
对 其 进行 修改 。 在 这 种 情况 下 ， 我 们 通常 想 在 新 版 本 里 调用 来 自 基 础 类 
的 方法 。 但 在 scrub0) 里 ， 不 可 只 是 简单 地 发 出 对 scrub0O 的 调用 。 那 样 便 




















造成 了 递归 调用 ， 我 们 不 愿 看 到 这 一 情况 。 为 解决 这 个 问题 ，Java 提 供 
了 一 个 super 关 键 字 ， 它 引用 当前 类 已 从 中 继承 的 一 个 “ 超 
类 ”(Superclass) 。 所 以 表达 式 super.scrub0 调 用 的 是 方法 scrub0 的 基础 


W 
类 版 本 





进行 继承 时 ， 我 们 并 不 限于 只 能 使 用 基础 类 的 方法 。 亦 可 在 衍生 出 来 的 
类 里 加 入 自己 的 新 方法 。 这 时 采取 的 做 法 与 在 普通 类 里 添加 其 他 任何 方 
法 是 完全 一 样 的 : 只 需 简 单 地 定义 它 即 可 。extends 关 键 字 提醒 我 们 准备 
和 
J— BPH 0 


在 Detergent.main() 里 ， 我 们 可 看 到 对 于 Detergent 对 象 ， 可 调用 Cleanser 
以 及 Detergent 内 所 有 可 用 的 方法 (如 foam())。 


6.2.1 初始 化 基础 类 


由 于 这 儿 涉 及 到 两 个 类 一 一 基础 类 及 生生 类 ， 而 不 再 是 以 前 的 一 个 ， 所 
以 在 想象 衍生 类 的 结果 对 象 时 ， 可 能 会 产生 一 些 迷 惑 。 从 外 部 看 ， 似 乎 
新 类 拥有 与 基础 类 相同 的 接口 ， 而 且 可 包含 一 些 和 额外 的 方法 和 字段 。 但 
继承 并 非 仅仅 简单 地 复制 基础 类 的 接口 了 事 。 创 建 衍生 类 的 一 个 对 象 
时 ， 它 在 其 中 包含 了 基础 类 的 一 个 “ 子 对 象 ?。 这 个 子 对 象 束 象 我 们 根据 
基础 类 本 里 创建 了 它 的 一 个 对 象 。 从 外 部 看 ， 基 础 类 的 子 对 象 已 封装 到 
衍生 类 的 对 象 里 了 。 


当然 ， 基 础 类 子 对 象 应 该 正确 地 初始 化 ， 而 且 只 有 一 种 方法 能 保证 这 一 
点 : 在 构建 器 中 执行 初始 化 ， 通 过 调用 基础 类 构建 器 ， 后 者 有 足够 的 能 
力 和 权限 来 执行 对 基础 类 的 初始 化 。 在 衍生 类 的 构建 器 中 ，Java 会 自动 
ee 。 下 面 这 个 例子 回 大 家 展示 了 对 这 种 三 级 继 
ZK HJM: 


//: Cartoon.java 

















// Constructor calls during inheritance 
class Art { 


Art() { 


System.out.println("Art constructor"); 


class Drawing extends Art { 
Drawing() { 


System.out.println("Drawing constructor"); 


} 


public class Cartoon extends Drawing { 
Cartoon() { 


System.out.printin("Cartoon constructor"); 


} 


public static void main(String[] args) { 


Cartoon x = new Cartoon(); 


} 
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该 程序 的 输出 显示 了 自动 调用 : 
Art constructor 
Drawing constructor 


Cartoon constructor 


可 以 看 出 ， 构 建 是 在 基础 类 的 “外 部 ”进行 的 ， 所 以 基础 类 会 在 衍生 类 访 
问 它 之 前 得 到 正确 的 初始 化 。 


即使 没有 为 Cartoon0) 创 建 一 个 构建 器 ， 编 译 器 也 会 为 我 们 目 动 合成 一 个 
默认 构建 器 ， 并 发 出 对 基础 类 构建 器 的 调用 。 

1. 含有 自 变 量 的 构建 器 

上 述 例 子 有 自己 默认 的 构建 占 ， 也 就 是 说 ， 它 们 不 含 任何 自 变 量 。 编 译 
器 可 以 很 容易 地 调用 它们 ， 因 为 不 存在 具体 传递 什么 自 变 量 的 问题 。 如 
果 类 没有 默认 的 自 变 量 ， 或 者 想 调用 含有 一 个 自 变 量 的 某 个 基础 类 构建 
器 ， 必 须 明 确 地 编写 对 基础 类 的 调用 代码 。 这 是 用 super 关 键 字 以 及 适当 
的 自 变 量 列表 实现 的 ， 如 下 所 示 : 


//: Chess.java 




















// Inheritance, constructors and arguments 
class Game { 
Game(int i) { 


System.out.println("Game constructor"); 


} 
class BoardGame extends Game { 
BoardGame(int i) { 
super(i); 


System.out.println("BoardGame constructor"); 


} 


public class Chess extends BoardGame { 
Chess() { 


super(11); 


System.out.println("Chess constructor"); 
public static void main(String[] args) { 
Chess x = new Chess(); 
Í 
} ///:~ 





如 果 不 调用 BoardGames0O 内 的 基础 类 构建 器 ， 编 译 器 束 会 报告 目 己 找 不 
到 Games0) 形 式 的 一 个 构建 器 。 除 此 以 外 ， 在 衍生 类 构建 器 中 ， 对 基础 
0 a 
BH) 。 


2. 捕获 基本 构建 器 的 违例 


正如 刚才 指出 的 那样 ， 编 译 器 会 强迫 我 们 在 衍生 类 构建 融 的 主体 中 首先 
设置 对 基础 类 构建 器 的 调用 。 这 意味 着 在 它 之 前 不 能 出 现任 何 东 西 。 正 
如 大 家 在 第 9 章 会 看 到 的 那样 ， 这 同时 也 会 防止 衍生 类 构建 器 捕获 来 自 
一 个 基础 类 的 任何 违例 事件 。 显 然 ， 这 有 时 会 为 我 们 造成 不 便 。 











6.3 合成 与 继承 的 结合 


许多 时 候 部 要 求 将 合成 与 继承 两 种 技术 结合 起 来 使 用 。 下 面 这 个 例子 展 
示 了 如 何 同时 采用 继承 与 合成 技术 ， 从 而 创建 一 个 更 复杂 的 类 ， 同 时 进 
行 必要 的 构建 器 初始 化 工作 : 


//: PlaceSetting.java 


// Combining composition & inheritance 
class Plate { 
Plate(int i) { 


System.out.println("Plate constructor"); 


J 


class DinnerPlate extends Plate { 
DinnerPlate(int i) { 
super (i); 
System.out.printin( 


"DinnerPlate constructor"); 


} 
class Utensil { 
Utensil(int i) { 


System.out.println( "Utensil constructor"); 


} 


class Spoon extends Utensil { 
Spoon(int i) { 
super(1); 


System.out.println( "Spoon constructor"); 


} 


class Fork extends Utensil { 
Fork(int i) { 
super(i); 


System.out.printin("Fork constructor"); 


} 


class Knife extends Utensil { 
Knife(int i) { 
super (i); 


System.out.println("Knife constructor"); 


} 


// A cultural way of doing something: 
class Custom { 
Custom(int i) { 


System.out.println( "Custom constructor"); 


} 


public class PlaceSetting extends Custom { 
Spoon sp; 
Fork frk; 
Knife kn; 
DinnerPlate pl; 
PlaceSetting(int i) { 
super(i + 1); 
sp = new Spoon(i + 2); 
frk = new Fork(i + 3); 


kn 


new Knife(i + 4); 
pl = new DinnerPlate(i + 5); 
System.out.printin( 
"PlaceSetting constructor"); 
} 
public static void main(String[] args) { 
PlaceSetting x = new PlaceSetting(9)j; 


} 
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尽管 编译 器 会 强迫 我 们 对 基础 类 进行 初始 化 ， 并 要 求 我 们 在 构建 器 最 开 
头 做 这 一 工作 ， 但 它 并 不 会 监视 我 们 是 否 正 确 初始 化 了 成 员 对 象 。 所 以 
对 此 必须 特别 加 以 留意 。 


6.3.1 确保 正确 的 清除 


Java 不 具备 象 C++ 的 “破坏 圳 ?那样 的 概念 。 在 C++ 中 ， 一 旦 破坏 〈 清 
BR) 一 个 对 象 ， 惑 会 目 动 调用 破坏 融 方 法 。 之 所 以 将 其 省 略 ， 大 概 是 由 
于 在 Java 中 只 再 简单 地 志 记 对 象 ， 不 需 强行 破坏 它们 。 世 圾 收集 霹 会 在 
必要 的 时 候 上 自动 回收 内 存 。 

垃圾 收集 器 大 多 数 时 候 都 能 很 好 地 工作 ， 但 在 某 些 情况 下 ， 我 们 的 类 可 
能 在 目 己 的 存在 时 期 采取 一 些 行动 ， 而 这 些 行动 要 求 必 须 进 行 明确 的 清 
除 工作 。 正 如 第 4 章 已 经 指出 的 那样 ， 我 们 并 不 知道 垃圾 收集 器 什么 时 
候 才 会 显 喘 ， 或 者 说 不 知 它 何 时 会 调用 。 所 以 一 旦 希望 为 一 个 类 清除 什 
么 东西 ， 必 须 写 一 个 特别 的 方法 ， 明 确 、 专 门 地 来 做 这 件 事情 。 同 时 ， 
还 要 让 客户 程序 员 知 道 他 们 必须 调用 这 个 方法 。 而 在 所 有 这 一 切 的 后 
面 ， 就 如 第 9 革 (违例 控制 要 详细 解释 的 那样 ， 必 须 将 这 样 的 清除 代 
码 置 于 一 个 finally 从 名 中， 从 而 防范 任何 可 能 出 现 的 违例 事件 。 


下 面 介绍 的 是 一 个 计算 机 辅助 设计 系统 的 例子 ， 它 能 在 屏幕 上 描绘 图 
形 : 


//: CADSystem. java 


// Ensuring proper cleanup 
import java.util.*; 
class Shape { 
Shape(int i) { 
System.out.println("Shape constructor"); 
J 
void cleanup() { 


System.out.println("Shape cleanup"); 


class Circle extends Shape { 
Circle(int i) { 
super(i); 
System.out.println("Drawing a Circle"); 
} 
void cleanup() { 
System.out.println("Erasing a Circle"); 


super.cleanup(); 


} 


class Triangle extends Shape { 
Triangle(int i) { 
super(i); 
System.out.println( "Drawing a Triangle"); 
} 
void cleanup() { 
System.out.println( "Erasing a Triangle"); 


super.cleanup(); 


j 


class Line extends Shape { 
private int start, end; 


Line(int start, int end) { 


super(start); 
this.start = start; 
this.end = end; 
System.out.println("Drawing a Line: " + 
start + ", " + end); 
} 
void cleanup() { 
System.out.println("Erasing a Line: " + 
start + ", " + end); 


super .cleanup(); 
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public class CADSystem extends Shape { 
private Circle c; 
private Triangle t; 
private Line[] lines = new Line[10]; 
CADSystem(int i) { 
super(i + 1); 
for(int j = 0; j < 10; j++) 
lines[j] = new Line(j, j*j); 


c = new Circle(1); 


t = new Triangle(1); 


System.out.println("Combined constructor"); 


} 
void cleanup() { 
System.out.printin("CADSystem.cleanup()"); 
t.cleanup(); 
c.cleanup(); 
for(int i = 0; i < lines.length; i++) 
lines[i].cleanup(); 
super.cleanup(); 
} 
public static void main(String[] args) { 
CADSystem x = new CADSystem(47); 
try { 
// Code and exception handling... 
} finally { 


x.cleanup(); 


} 
} ///:~ 





这 个 系统 中 的 所 有 东西 都 属于 某 种 Shape (几何 形状 〉 。Shape 本 身 是 一 
种 Object( 对 象 )， 因 为 它 是 从 根 类 明确 继承 的 。 每 个 类 都 重新 定义 了 
Shape 的 deanup0 方 法 ， 同 时 还 要 用 super 调 用 那个 方法 的 基础 类 版 本 。 

尽管 对 象 存 在 期 间 调 用 的 所 有 方法 都 可 负责 做 一 些 要 求 清除 的 工作 ， 但 
对 于 特定 的 Shape 类 一 一 Circle 〈 圆 ) ~ Triangle (三 角形 ) 以 及 Line (A 
线 ) ， 它 们 都 拥有 上 自己 的 构建 器 ， 能 完成 * 作 网 ”(draw) 任务 。 每 个 类 














RN aa 
前 的 景象 。 


在 main() 中 ， 可 看 到 两 个 新 关键 字 : try 和 finaly。 我 们 要 到 第 9 章 才 会 回 
大 家 正式 引荐 它们 。 其 中 ，try 关 键 字 指出 后 面 跟随 的 块 〈 由 人 花 括 号 定 
FL) 是 一 个 “警戒 区 ”。 也 就 是 说 ， 它 会 受到 特别 的 待遇 。 其 中 一 种 符 遇 
就 是 : 该 警戒 区 后 面 跟随 的 finally 从 名 的 代码 肯定 会 得 以 执行 一 一 不 管 
try 块 到 底 存 不 存在 (通过 违例 控制 技术 ，try 块 可 有 多 种 不 寻常 的 应 
H) 。 在 这 里 ，finally 从 句 的 意思 是 “总 是 为 x 调用 dleanup()， 无 论 会 及 
生 什 么 事情 ?”。 这 些 关 键 字 将 在 第 9 章 进 行 全 面 、 完 整 的 解释 。 


在 目 己 的 清除 方法 中 ， 必 须 注意 对 基础 类 以 及 成 员 对 象 清除 方法 的 调用 
顺序 一 一 假 徊 一 个 子 对 象 要 以 另 一 个 为 基础 。 通 常 ， 应 采取 与 C++ 编译 
器 对 它 的 “破坏 器 ?采取 的 同样 的 形式 : 首先 完成 与 类 有 关 的 所 有 特殊 工 
作 〈 可 能 要 求 基 础 类 元 素 仍然 可 见 ) ， 然 后 调用 基础 类 清除 方法 ， 就 象 
这 儿 演 示 的 那样 。 


许多 情况 下 ， 清 除 可 能 并 不 是 个 问题 只 需 让 垃圾 收集 器 尽 它 的 职责 即 
可 。 但 一 旦 必须 由 目 己 明确 清除 ， 就 必须 特别 谨慎 ， 并 要 求 周 全 的 考 
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1. 垃圾 收集 的 顺序 


不 能 指望 自己 能 确切 知道 何 时 会 开始 垃圾 收集 。 垃 圾 收集 器 可 能 永远 不 
会 得 到 调用 。 即 使 得 到 调用 ， 它 也 可 能 以 自己 愿意 的 任何 顺序 回收 对 
象 。 除 此 以 外 ，Java 1.0 实 现 的 垃圾 收集 器 机 制 通 常 不 会 调用 finalize() 方 
法 。 除 内 存 的 回收 以 外 ， 其 他 任何 东西 都 最 好 不 要 依赖 垃圾 收集 器 进行 
回收 。 若 想 明 确 地 清除 什么 ， 请 制作 自己 的 清除 方法 ， 而 且 不 要 依赖 
finalize()。 然 而 正如 以 前 指出 的 那样 ， 可 强迫 Javal.1 调 用 所 有 收尾 模块 


(Finalizer) 。 















































6.3.2 名 字 的 隐藏 


只 有 C++ 程序 员 可 能 才 会 惊讶 于 名 字 的 隐藏 ， 因 为 它 的 工作 原理 与 在 
C++ 里 是 完全 不 同 的 。 如 果 Java 基 础 类 有 一 个 方法 名 被 <“ 过载” 使 用 多 
次 ， 在 衍生 类 里 对 那个 方法 名 的 重新 定义 就 不 会 隐藏 任何 基础 类 的 版 
本 。 所 以 无 论 方法 在 这 一 级 还 是 在 一 个 基础 类 中 定义 ， 过 载 都 会 生效 : 








//: Hide.java 


// Overloading a base-class method name 
// in a derived class does not hide the 
// base-class versions 
class Homer { 
char doh(char c) { 
System.out.printin("doh(char)"); 
return ‘d'; 
} 
float doh(float f) { 
System.out.println("doh(float)"); 


return 1.0f; 


} 
Class Milhouse {} 
class Bart extends Homer { 
void doh(Milhouse m) {} 
} 
class Hide { 
public static void main(String[] args) { 
Bart b = new Bart(); 


b.doh(1); // doh(float) used 


b.doh('x'); 
b.doh(1.0f); 
b.doh(new Milhouse()); 


} 
} ///:~ 


正如 下 一 章 会 讲 到 的 那样 ， 很 少 会 用 与 基础 类 里 完全 一 致 的 签名 和 返回 
类 型 来 覆盖 同名 的 方法 ， 否 则 会 使 人 感到 迷惑 〈 这 正 是 C++ 不 允许 那样 
做 的 原因 ， 所 以 能 够 防止 产生 一 些 不 必要 的 错误 ) 。 


6.4 到 撒 选 择 合成 还 是 继承 


fan 成 还 是 继承 ， 痢 允许 我 们 将 子 对 象 置 于 自己 的 新 类 中 。 大 家 或 许 
奇怪 两 者 间 的 差 寞 ， 以 及 到 底 该 如 何 选 择 。 


如 果 想 利用 新 类 内 部 一 个 现 有 类 的 特性 ， 而 个 想 使 用 它 的 接口 ， 通常 应 
选择 合成 。 也 就 是 说 ， 我 们 可 舱 入 一 个 对 象 ， 使 自己 能 用 它 实 现 新 类 的 
特性 。 但 新 类 的 用 户 会 看 到 我 们 已 定义 的 接口 ， 而 不 是 来 自 赔 入 对 象 的 
接口 。 考 虑 到 这 种 效果 ， 我 们 需 在 新 类 里 磐 入 现 有 类 的 private 对 象 。 
有 些 时 候 ， 我 们 想 让 类 用 户 直 接 访 问 新 类 的 合成 。 也 就 是 说 ， 需 要 将 成 
员 对 象 的 属性 变 为 public。 成 员 对 象 会 将 自身 隐藏 起 来 ， 所 以 这 是 一 种 
安全 的 做 法 。 而 且 在 用 户 知道 我 们 准备 合成 一 系列 组 件 时 ， 接 口 束 更 容 
Sy Fah. car GA) 对 象 便 是 一 个 很 好 的 例子 : 


//: Car.java 








// Composition with public objects 
class Engine { 

public void start() {} 

public void rev() {} 

public void stop() {} 
} 
class Wheel { 

public void inflate(int psi) {} 
} 
class Window { 


public void rollup() {} 


public void rolldown() {} 
} 
class Door { 
public Window window = new Window(); 
public void open() {} 
public void close() {} 
} 
public class Car { 


public Engine engine = new Engine(); 


public Wheel[] wheel = new Wheel[4]; 
public Door left = new Door(), 
right = new Door(); // 2-door 
Car() { 
for(int i = 0; i < 4; i++) 
wheel[i] = new Wheel(); 
} 
public static void main(String[] args) { 
Car car = new Car(); 
car.left.window.rollup(); 


car.wheel[0].inflate(72); 


} ///:~ 


由 于 汽车 的 装配 是 故障 分 析 时 需要 考虑 的 一 项 因 系 (并 非 只 是 基础 设计 
简单 的 一 部 分 ) ， 所 以 有 助 于 客户 程序 员 理 解 如 何 使 用 类 ， 而 且 类 创建 
者 的 编程 复杂 程度 也 会 大 幅度 降低 。 


如 选择 继承 ， 就 需要 取得 一 个 现成 的 类 ， 并 制作 它 的 一 个 特殊 版 本 。 通 
常 ， 这 意味 看 我 们 准备 使 用 一 个 第 规 用 途 的 类 ， 并 根据 特定 的 需求 对 其 
进行 定制 。 只 需 和 加 想象 ， 束 知道 自己 不 能 用 一 个 车 辆 对 象 来 合成 一 辆 
汽车 一 一 汽车 并 不 “包含 ”车辆 ， 相 反 ， 它 “属于 ”车 辆 的 一 种 类 别 。“ 属 
于 ”关系 是 用 继承 来 表达 的 ， 而 “包含 ”关系 是 用 合成 来 表达 的 。 

















6.5 protected 





我 们 已 理解 了 继承 的 概念 ，protected 这 个 关键 字 最 后 终于 有 了 意 

。 在 理想 情况 下 ，Pprivate 成 员 随 时 都 是 “私有 ”的 ， 任 何人 不 得 访问 。 
ETSE, AAEE Eb 东西 深 深 地 藏 起 来 ， 但 同时 允许 访问 衍 
生 类 的 成 员 。protected 关 键 字 可 帮助 我 们 做 到 这 一 点 。 它 的 意思 是 “ 它 本 
身 是 私有 的 ， 但 可 由 从 这 个 类 继承 的 任何 东西 或 者 同一 个 包 内 的 其 他 任 
何 东 西 访问 *”。 也 就 是 说 ，Java 中 的 protected 会 成 为 进入 “友好 ”状态 。 
我 们 采取 的 最 好 的 做 法 是 保持 成 员 的 private 状 态 论 如 何 都 应 保留 
对 基 础 的 实施 细节 进行 修改 的 权利 。 在 这 一 前 提 下 ， 可 通过 protected 方 
法 允许 类 的 继承 者 进行 受到 控制 的 访问 : 


//: Orc.java 























// The protected keyword 

import java.util.*; 

class Villain { 
private int i; 
protected int read() { return i; } 
protected void set(int ii) { i = ii; } 
public Villain(int ii) { i = ii; } 
public int value(int m) { return m*i; } 

} 

public class Orc extends Villain { 
private int j; 
public Orc(int jj) { super(jj); j = jj; } 


public void change(int x) { set(x); } 


} ///:~ 


可 以 看 到 ，change() 拥 有 对 set() 的 访问 权限 ， 因 为 它 的 属性 是 
protected (受到 保护 的 〉。 


6.6 RIEF E 


继承 的 一 个 好 处 是 它 文 持 “ 昧 积 开 发 >， 多 许 我 们 引入 新 的 代码 ， 同 时 不 
会 为 现 有 代码 造成 错误 。 这 样 可 将 新 错误 隔离 到 新 代码 里 。 通 过 从 一 个 
现成 的 、 功 能 性 的 类 继承 ， 同 时 增添 成 员 新 的 数据 成 员 及 方法 〈 并 重新 
定义 现 有 方法 ) ， 我 们 可 保持 现 有 代码 原封 不 动 〈 另 外 有 人 也 许 仍 在 使 
用 它 ) ， 不 会 为 其 引入 目 己 的 编程 错误 。 一 旦 出 现 错误 ， 就 知道 它 肯 定 
是 由 于 自己 的 新 代码 造成 的 。 这 样 一 来 ， 与 修改 现 有 代码 的 主体 相 比 ， 
改正 错误 所 需 的 时 间 和 精力 就 可 以 少 很 多 。 


类 的 隔离 效果 非常 好 ， 这 是 许多 程序 员 事 先 没 有 预料 到 的 。 甚 至 不 需要 
方法 的 源 代码 来 实现 代码 的 再 生 。 最 多 只 需要 导入 一 个 包 ( 这 对 于 继承 
和 合并 都 是 成 并 的 〉。 


大 家 要 记 住 这 样 一 个 重点 : 程序 开发 是 一 个 不 断 递增 或 者 素 积 的 过 程 ， 
就 象 人 们 学 习 知 识 一 样 。 当 然 可 根据 要 求 进行 尽 可 能 多 的 分 析 ， 但 在 一 
个 项 目的 设计 之 初 ， 谁 都 不 可 能 提前 获知 所 有 的 答案 。 如 采 能 将 目 己 的 
项 目 看 作 一 个 有 机 的 、 能 不 断 进步 的 生物 ， 从 而 不 断 地 发 展 和 改进 它 ， 
就 有 望 获得 更 大 的 成 功 以 及 更 直接 的 反馈 。 


尽管 继承 是 一 种 非常 有 用 的 技术 ， 但 在 东 些 情况 下 ， 特 别 是 在 项 目 稳 定 
下 来 以 后 ， 仍 然 需 要 从 新 的 角度 考察 自己 的 类 结构 ， 将 其 收缩 成 一 个 更 
灵活 的 结构 。 请 记 住 ， 继 承 是 对 一 种 特殊 关系 的 表达 ， 意 味 痢 “这 个 新 
类 属于 那个 旧 类 的 一 种 类 型 *。 我 们 的 程序 不 应 纠缠 于 一 些 细 树 末节 ， 
而 应 着 眼 于 创建 和 操作 各 种 类 型 的 对 象 ， 用 它们 表达 出 来 自 “ 问 题 空 
间 ” 的 一 个 模型 。 



































6.7 Eid AY 


继承 最 值得 注意 的 地 方 就 是 它 没 有 为 新 类 提供 方法 。 继 承 是 对 新 类 和 基 
有 
一 不 类 型 ”。 


这 种 表达 并 不 仅仅 是 对 继承 的 一 种 形象 化 解释 ， 继 承 是 直接 由 语言 提供 
文 持 的 。 作 为 一 个 例子 ， 大 家 可 考虑 一 个 名 为 Instrument 的 基础 类 ， 它 
用 于 表示 乐器 ;， 男 一 个 衍生 类 叫 作 Wind。 由 于 继承 意味 着 基础 类 的 所 
有 方法 亦 可 在 衍生 出 来 的 类 中 使 用 ， 所 以 我 们 发 给 基础 类 的 任何 消息 
可 发 给 衍生 类 。 知 mstrument 类 有 一 个 play(0) 方 法 ， 则 Wind 设 备 也 会 有 这 
个 方法 。 这 意味 着 我 们 能 肯定 地 认为 一 个 Wind 对 象 也 是 Instrument 的 一 
种 类 型 。 下 面 这 个 例子 揭示 出 编译 器 如 何 提供 对 这 一 概念 的 文 持 : 


//: Wind.java 














// Inheritance & upcasting 
import java.util.*; 
class Instrument { 
public void play() {} 
static void tune(Instrument i) { 
a ee 


i.play(); 


} 


// Wind objects are instruments 
// because they have the same interface: 


Class Wind extends Instrument { 


public static void main(String[] args) { 
Wind flute = new Wind(); 
Instrument.tune(flute); // Upcasting 
} 
} ///:~ 





这 个 例子 中 最 有 趣 的 无 疑 是 tune() 方 法 ， 它 能 接受 一 个 Instrument 人 句柄 。 
但 在 Wind.main() 中 ，tune0 方 法 是 通过 为 其 赋予 一 个 Wind 句 柄 来 调用 

的 。 由 于 Java 对 类 型 检查 特别 严格 ， 所 以 大 家 可 能 会 感到 很 奇怪 ， 为 什 
么 接收 一 种 类 型 的 方法 也 能 接收 另 一 种 类 型 呢 ? 但 是 ， 我 们 一 定 要 认识 
到 一 个 Wind 对 象 也 是 一 个 Instrument 对 象 。 而 且 对 于 不 在 Wind 中 的 一 个 
Instrument (Rat) ， 没 有 方法 可 以 由 tune0 调 用 。 在 tune0 中 ， 代 码 适 用 
于 Instrument 以 及 从 Instrument 衍 生出 来 的 任何 东西 。 在 这 里 ， 我 们 将 从 
一 个 Wind 句 柄 转换 成 一 个 Instrument 句 柄 的 行为 叫 作 * 上 漳 造 型 ”。 


6.7.1 HNA EWE”? 


之 所 以 叫 作 这 个 名 字 ， 除 了 有 一 定 的 历史 原因 外 ， 也 是 由 于 在 传统 意义 
上 ， 关 继承 图 的 国法 是 根 位 于 最 顶部 ， 再 逐渐 问 下 扩展 《〈“ 当 然 ， 可 根据 
自己 的 习惯 用 任何 方法 描绘 这 种 图 ) 。 因 和 素 ，Wind.java 的 继承 图 就 象 下 
面 这 个 样子 : 














由 于 造型 的 方向 是 从 衍生 类 到 基础 类 ， 箭 头 朝 上 ， 上 所 以 通常 把 它 叫 

作 “ 上 渊 造型 ”， 即 Upcasting。 上 滑 造 型 肯定 是 安全 的 ， 因 为 我 们 是 从 一 
个 更 特殊 的 类 型 到 一 个 更 常规 的 类 型 。 换 言 之 ， 衍 生 类 是 基础 类 的 一 个 
超 集 。 它 可 以 包含 比 基 础 类 更 多 的 方法 ， 但 它 至 少 包 含 了 基础 类 的 方 
法 。 进 行 上 漳 造 型 的 时 候 ， 类 接口 可 能 出 现 的 唯一 一 个 问题 是 它 可 能 
失 方 法 ， 而 不 是 局 得 这 些 方法 。 这 便 是 在 没有 任何 明确 的 造型 或 者 其 他 











特殊 标注 的 情况 下 ， 编 译 器 为 什么 允许 上 济 造 型 的 原因 所 在 。 
也 可 以 执行 下 调 造 型 ， 但 这 时 会 面临 第 11 章 要 详细 讲述 的 一 种 困境 。 
1. 再 论 合成 与 继承 


在 面 问 对 象 的 程序 设计 中 ， 创 建 和 使 用 代码 最 可 能 采取 的 一 种 做 法 走 : 
将 数据 和 方法 统一 封 沪 到 一 个 类 里 ， 并 且 使 用 那个 类 的 对 象 。 有 些 时 

候 ， 需 通过 “合成 ?技术 用 现成 的 类 来 构造 新 类 。 而 继承 是 最 少见 的 一 种 
做 法 。 因 上 此， 尽管 继承 在 学 习 OOP 的 过 程 中 得 到 了 大 量 的 强调 ， 但 并 不 
意味 着 应 该 尽 可 能 地 到 处 使 用 它 。 相 反 ， 使 用 它 时 要 特别 慎重 。 只 有 在 
清楚 知道 继承 在 所 有 方法 中 最 有 效 的 前 担 下 ， 才 可 考虑 它 。 为 判断 目 己 
到 压 应 该 选用 合成 还 是 继承 ， 一 个 最 简单 的 办 法 就 是 考虑 是 人 否 需 要 从 新 
类 上 调 造 型 回 基 础 类。 符 必 须 上 溯 ， 就 需要 继承 。 但 如 果 不 需要 上 调 造 
型 ， 就 应 提醒 自己 防止 继承 的 小 用。 在 下 一 章 里 《多 形 性 ) ， 会 向 大 家 
介绍 必须 进行 上 调 造 型 的 一 种 场合 。 但 只 要 记 住 经 党 问 目 己 “我 真 的 需 
要 上 渊 造型 吗 ”， 对 于 合成 还 是 继承 的 选择 就 不 应 该 是 个 太 大 的 问题 。 


























6.8 final 关 键 字 


由 于 语 境 〈 应 用 环境 ) 不 同 ，final 关 键 字 的 含义 可 能 会 稍微 产生 一 些 关 
异 。 但 它 最 一 般 的 意思 就 是 声明 “这 个 东西 不 能 改变 ”。 之 所 以 要 茶 止 改 
变 ， 可 能 是 考虑 到 两 方面 的 因 系 : 设计 或 效率 。 由 于 这 两 个 原因 颇 有 些 
区 别 ， 所 以 也 许 会 造成 fnal 关 键 字 的 误 用 。 


在 接 下 去 的 小 节 里 ， 我 们 将 讨论 final 关 键 字 的 三 种 应 用 场合 : 数据 、 方 
法 以 及 类 。 


6.8.1 final 数 据 


许多 程序 设计 语言 都 有 目 己 的 办 法 告诉 编译 器 菏 个 数据 是 “种 数 "。 钟 数 
主要 应 用 于 下 述 两 个 方面 : 


(1) 编译 期 常数 ， 它 永远 不 会 改变 
(2) 在 运行 期 初始 化 的 一 个 值 ， 我 们 不 希望 它 发 生变 化 


对 于 编译 期 的 和 常数， 编译 器 (程序) 可 将 常数 值 “ 封 装 ” 到 需要 的 计算 过 
程 里 。 也 就 是 说 ， 计 算 可 在 编译 期 间 提 前 执行 ， 从 而 节省 运行 时 的 一 些 
开销 。 在 Java 中 ， 这 些 形式 的 常数 必须 属于 基本 数据 类 型 

(Primitives) ， 而 且 要 用 final 关 键 字 进行 表达 。 在 对 这 样 的 一 个 常数 进 
行 定义 的 时 候 ， 必 须 给 出 一 个 值 。 


无 论 static 还 是 final 字 段 ， 都 只 能 存储 一 个 数据 ， 而 且 不 得 改变 。 


石 随同 对 象 句 柄 使 用 final， 而 不 是 基本 数据 类 型 ， 它 的 含义 就 稍微 让 人 
有 点 儿 迷 糊 了 。 对 于 基本 数据 类 型 ，final 会 将 值 变 成 一 个 常数 ， 但 对 于 
对 象 句柄 ，final 会 将 句柄 变 成 一 个 常数 。 进 行 声明 时 ， 必 须 将 句柄 初始 
化 到 一 个 具体 的 对 象 。 而 且 永 远 不 能 将 句柄 变 成 指向 力 一 个 对 象 。 然 
而 ， 对 象 本 身 是 可 以 修改 的 。Java 对 此 未 提供 任何 手段 ， 可 将 一 个 对 象 
直接 变 成 一 个 第 数 〈 但 是 ， 我 们 可 自己 编写 一 个 类 ， 使 其 中 的 对 象 具 
有 “常数 ”效果 ) 。 这 一 限制 也 适用 于 数组 ， 它 也 属于 对 象 。 


下 面 是 演示 final 字 段 用 法 的 一 个 例子 : 


























//: FinalData.java 


// The effect of final on fields 
class Value { 
int i = 1; 
} 
public class FinalData { 
// Can be compile-time constants 
final int il = 9; 
static final int I2 = 99; 
// Typical public constant: 
public static final int I3 = 39; 
// Cannot be compile-time constants: 


final int 14 = (int)(Math.random()*20); 


static final int i5 = (int)(Math.random()*20); 


Value vi = new Value(); 

final Value v2 = new Value(); 

static final Value v3 = new Value(); 

//! final Value v4; // Pre-Java 1.1 Error: 
// no initializer 
// Arrays: 


final int[] a= { 1, 2, 3, 4, 5, 6}; 


public void print(String id) { 


System.out.printin( 
id + Wns " + "ig = " + i4 + 


人 ae a wab 


public static void main(String[] args) { 


FinalData fd1 = new FinalData(); 
//! fd1.i1++; // Error: can't change value 
fdi.v2.it++; // Object isn't constant! 


fdi.vi = new Value(); // OK -- not final 


for(int i = 0; i < fdi.a.length; i++) 


// | 


// | 


// | 


fdi.a[i]++; // Object isn't constant! 
fdi.v2 = new Value(); // Error: Can't 
fdi.v3 = new Value(); // change handle 
fdi.a = new int[3]; 
fdi.print("fdi"); 
System.out.printin("Creating new FinalData"); 
FinalData fd2 = new FinalData(); 
fdi.print("fdi"); 


fd2.print("fd2"); 


} ///:~ 


由 于 计 和 ZE 都 是 具有 final 属 性 的 基本 数据 类 型 ， 并 含有 编译 期 的 值 ， 所 
以 它们 除了 能 作为 编译 期 的 常数 使 用 外 ， 在 任何 导入 方式 中 也 不 会 出 现 
任何 不 同 。I3 是 我 们 体验 此 类 和 常数 定 义 时 更 典型 的 一 种 方式 : public 表 
示 它 们 可 在 包 外 使 用 :Static 强调 它们 只 有 一 个 ;而 final 表 明 它 是 一 个 销 
数 。 注 意 对 于 含有 固定 初始 化 值 〈 即 编译 期 常数 ) 的 fianl static 基 本 数据 
类 型 ， 它 们 的 名 字 根 据 规 则 要 全 部 采用 大 写 。 也 要 注意 i5 在 编译 期 间 是 
未 知 的 ， 所 以 它 没 有 大 写 。 


不 能 由 于 人 菏 样 东西 的 属性 是 final， 束 认定 它 的 值 能 在 编译 时 期 知道 。i4 
和 i5 向 大 家 证 明了 这 一 点 。 它 们 在 运行 期 间 使 用 随机 生成 的 数学 。 例 子 
的 这 一 部 分 也 同 大 家 揭示 出 将 final 值 设 为 static 和 非 static 之 间 的 过 异 。 只 
有 当 值 在 运行 期 间 初 始 化 的 前 担 下 ， 这 种 差异 才 会 揭示 出 来 。 因 为 编译 
期 间 的 值 被 编译 器 认为 是 相同 的 。 这 种 差异 可 从 输出 结果 中 看 出 : 


fd1: i4 = 15, i5 = 9 























Creating new FinalData 
fd1: i4 = 15, i5 = 9 


fd2: i4 = 10, i5 = 9 


注意 对 于 fd1 和 fd2 来 说 ，i4 的 值 是 唯一 的 ， 但 i5 的 值 不 会 由 于 创建 了 男 
一 个 FinalData 对 象 而 发 生 改 变 。 那 是 因为 它 的 属性 是 static， 而 且 在 载 入 
时 初始 化 ， 而 非 每 创建 一 个 对 象 时 初始 化 。 


从 v1 到 v4 的 变量 铝 我 们 揭示 出 final 句 柄 的 含义 。 正 如 大 家 在 main0 中 看 
到 的 那样 ， 并 不 能 认为 由 于 v2 属 于 final， 所 以 就 不 能 再 改变 它 的 值 。 然 
而 ， 我 们 确实 不 能 再 将 v2 绑 定 到 一 个 新 对 象 ， 因 为 它 的 属性 是 final。 这 
便 是 final 对 于 一 个 句柄 的 确切 含义 。 我 们 会 发 现 同样 的 售 义 亦 适用 于 数 
组 ， 后 者 只 不 过 是 另 一 种 类 型 的 句柄 而 已 。 将 句柄 变 成 final 看 起 来 似乎 
不 如 将 基本 数据 类 型 变 成 final 那 么 有 用 。 


2. 空白 final 


Java 1.1 人 允许 我 们 创建 “空白 final*， 它 们 属于 一 些 特殊 的 字段 。 尽 管 被 声 
明成 fnal， 但 却 未 得 到 一 个 初始 值 。 无 论 在 哪 种 情况 下 ， 空 白 final 都 必 








须 在 实际 使 用 前 得 到 正确 的 初始 化 。 而 且 编 译 器 会 主动 保证 这 一 规定 得 
以 人 贯彻。 然而， 对 于 final 关 键 字 的 各 种 应 用 ， 空 白 final 具 有 最 大 的 灵活 
性 。 举 个 例子 来 说 ， 位 于 类 内 部 的 一 个 final 字 段 现 在 对 每 个 对 象 都 可 以 
有 上 所 不 同 ， 同 时 依然 保持 其 “不 变 ” 的 本 质 。 下 面 列 出 一 个 例子 : 


//: BlankFinal.java 











// "Blank" final data members 
class Poppet { } 
class BlankFinal { 
final int i = 0; // Initialized final 
final int j; // Blank final 
final Poppet p; // Blank final handle 
// Blank finals MUST be initialized 
// in the constructor: 
BlankFinal() { 


1; // Initialize blank final 


j 


p 
} 


new Poppet(); 


BlankFinal(int x) { 


j = x; // Initialize blank final 


p 
I 


new Poppet(); 


public static void main(String[] args) { 


BlankFinal bf = new BlankFinal(); 


} 
} ///:~ 





现在 强行 要 求 我 们 对 final 进 行 赋值 处 理 要 么 在 定义 字段 时 使 用 一 个 
表达 式 ， 要 么 在 每 个 构建 右 中 。 这 样 束 可 以 确保 final 字 上 段 在 使 用 前 获得 
正确 的 初始 化 。 

3. final A E 

Java 1.1 人 允许 我 们 将 自 变量 设 成 final 属 性 ， 方 法 是 在 自 变 量 列表 中 对 它们 
进行 适当 的 声明 。 这 意味 着 在 一 个 方法 的 内 部 ， 我 们 不 能 改变 目 变 量 句 
柄 指 回 的 东西 。 如 下 所 示 : 


//: FinalArguments.java 











// Using "final" with method arguments 
class Gizmo { 
public void spin() {} 
} 
public class FinalArguments { 
void with(final Gizmo g) { 
//! g = new Gizmo(); // Illegal -- g is final 
g.spin(); 
} 
void without(Gizmo g) { 
g = new Gizmo(); // OK -- g not final 


g.spin(); 


// void f(final int i) { i++; } // Can't change 
// You can only read from a final primitive: 
int g(final int i) { return i+ 1; } 
public static void main(String[] args) { 
FinalArguments bf = new FinalArguments(); 
bf.without(null); 
bf .with(null); 


} 
} ///:~ 


注意 此 时 仍然 能 为 final 自 变量 分 配 一 个 null( 空 ) 句 顶 ， 同 时 编译 器 不 
会 捕获 它 。 这 与 我 们 对 非 final 自 变量 采取 的 操作 是 一 样 的 。 


方法 f 和 和 8gO 向 我 们 展示 出 基本 类 型 的 自 变量 为 final 时 会 发 生 什么 情况 : 
我 们 只 能 读 取 自 变量 ， 不 可 改变 它 。 


6.8.2 final 方 法 


之 所 以 要 使 用 final 方 法 ， 可 能 是 出 于 对 两 方面 理由 的 考虑 。 第 一 个 是 为 
方法 “上 锁 ”， 防 止 任何 继承 类 改变 它 的 本 来 含义 。 设 计 程 序 时 ， 厦 希望 
一 个 方法 的 行为 在 继承 期 间 保 持 不 变 ， 而 且 不 可 被 禾 瘟 或 改写 ， 就 可 以 
采取 这 种 做 法 。 


采用 final 方 法 的 第 二 个 理由 是 程序 执行 的 效率 。 将 一 个 方法 设 成 final 

后 ， 编 译 侣 就 可 以 把 对 那个 方法 的 所 有 调用 都 置 入 “ 供 入 ”调用 里 。 只 要 
编译 器 发 现 一 个 final 方 法 调用 ， 葡 会 《根据 它 目 己 的 判断 ) 忽略 为 执行 
方法 调用 机 制 而 采取 的 常规 代码 插入 方法 (将 上 自 变 量 压 入 堆栈 ; 跳 至 方 
法 代码 并 执行 它 ， 跳 回来 ， 清除 堆栈 上 自 变 量 ， 最 后 对 返回 值 进 行 处 

HD 。 相 反 ， 它 会 用 方法 主体 内 实际 代码 的 一 个 副本 来 将 换 方 法 调用 。 
这 样 做 可 避免 方法 调用 时 的 系统 开销 。 当 然 ， 大 方法 体积 太 大 ， 那 么 程 
序 也 会 变 得 比 肿 ， 可 能 受到 到 不 到 众 入 代码 所 带 来 的 任何 性 能 提升 。 因 
为 任何 提升 都 被 花 在 方法 内 部 的 时 间 抵 消 了 。Java 编 译 占 能 目 动 侦 测 这 
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不 要 完全 相信 编译 器 能 正确 地 作出 所 有 判断 。 通 常 ， 只 有 在 方法 的 代码 
ze a a 
为 final 。 


类 内 所 有 private 方 法 都 自动 成 为 final。 由 于 我 们 不 能 访问 一 个 private 方 
法 ， 所 以 它 绝 对 不 会 被 其 他 方法 宪 盖 〈 共 强 行 这 样 做 ， 编 译 器 会 给 出 错 
误 提示 ) 。 可 为 一 个 private 方 法 添加 final 指 示 符 ， 但 却 不 能 为 那个 方法 
提供 任何 额外 的 含义 。 


6.8.3 final 类 


如 果 说 整个 类 都 是 final (在 它 的 定义 前 冠 以 final 关 键 字 〉 ， 就 表明 目 己 
不 希望 从 这 个 类 继承 ， 或 者 不 允许 其 他 任何 人 采取 这 种 操作 。 换 言 之 ， 
出 于 这 样 或 那样 的 原因 ， 我 们 的 类 肯定 不 需要 进行 任何 改变 ; 或 者 出 于 
安全 方面 的 理由 ， 我 们 不 希望 进行 子 类 化 〈 子 类 处 理 ) 。 


除 此 以 外 ， 我 们 或 许 还 考虑 到 执行 效率 的 问题 ， 并 想 确 保 涉 及 这 个 类 各 
对 象 的 所 有 行动 都 要 尽 可 能 地 有 效 。 如 下 所 示 : 


//: Jurassic.java 











// Making an entire class final 
class SmallBrain {} 
final class Dinosaur { 
int i = 7; 
int j = 1; 
SmallBrain x = new SmallBrain(); 
void f() {} 
} 


//! class Further extends Dinosaur {} 


// error: Cannot extend final class 'Dinosaur' 
public class Jurassic { 
public static void main(String[] args) { 
Dinosaur n = new Dinosaur(); 
n.f(); 
n.i = 40; 
n.j+t+; 
} 
} ///:~ 


注意 数据 成 员 既 可 以 是 final， 也 可 以 不 是 ， 取 决 于 我 们 具体 选择 。 应 用 
于 final 的 规则 同样 适用 于 数据 成 员 ， 无 论 类 是 否 补 定义 成 final。 将 类 定 
义 成 final 后 ， 结 果 只 是 禁止 进行 继承 一 一 没有 更 多 的 限制 。 然 而 ， 由 于 
它 禁 止 了 继承 ， 所 以 一 个 final 类 中 的 所 有 方法 都 默认 为 final。 因 为 此 时 
再 也 无 法 履 兰 它们。 所 以 与 我 们 将 一 个 方法 明确 声明 为 final 一 样 ， 编 译 
虱 此 时 有 相同 的 效率 选择 。 


可 为 final 类 内 的 一 个 方法 添加 final 指 示 符 ， 但 这 样 做 没有 任何 意义 。 
6.8.4 final 的 注意 事项 

设计 一 个 类 时 ， 往 往 需要 考虑 是 否 将 一 个 方法 设 为 final。 可 能 会 党 得 使 
用 目 己 的 类 时 执行 效率 非常 重要 ， 没 有 人 想 履 盖 上 自己 的 方法 。 这 种 想法 
在 某 些 时 候 是 正确 的 。 


但 要 慎重 作出 自己 的 假定 。 通 常 ， 我 们 很 难 预 测 一 个 类 以 后 会 以 什么 样 
的 形式 再 生 或 重复 利用 。 常 规 用 途 的 类 尤其 如 此 。 大 将 一 个 方法 定义 成 
final， 束 可 能 杜绝 了 在 其 他 程序 员 的 项 目 中 对 自己 的 类 进行 继承 的 途 
径 ， 因 为 我 们 根本 没有 想到 它 会 象 那 样 使 用 。 


标准 Java 库 是 阐述 这 一 观点 的 最 好 例子 。 其 中 特别 第 用 的 一 个 类 是 
Vector。 如 果 我 们 考虑 代码 的 执行 效率 ， 就 会 发 现 只 有 不 把 任何 方法 设 














为 final， 才 能 使 其 发 挥 更 大 的 作用 。 我 们 很 容易 束 会 想到 自己 应 继承 和 
履 盖 如 此 有 用 的 一 个 类 ， 但 它 的 设计 者 却 否 定 了 我 们 的 想法 。 但 我 们 至 
少 可 以 用 两 个 理由 来 反 驱 人 他们。 首先，Stack《〈 推 栈 ) 是 从 Vector 继承 来 
的 ， 亦 即 Stack“ 是 ”一 个 Vector， 这 种 说 法 是 不 确切 的 。 其 次 ， 对 于 
Vector 许多 重要 的 方法 ， 如 addElementO 以 及 elementAt0 等 ， 它 们 都 变 成 
J synchronized (同步 的 ) 。 正 如 在 第 14 章 要 讲 到 的 那样 ， 这 会 造成 显 
著 的 性 能 开销 ， 可 能 会 把 final 提 供 的 性 能 改善 抵 销 得 一 干 二 净 。 因 此 ， 
程序 员 不 得 不 猜测 到 底 应 该 在 哪里 进行 优化 。 在 标准 库 里 居然 采用 了 如 
此 笨拙 的 设计 ， 真 不 敢 想 象 会 在 程序 员 里 引发 什么 样 的 情绪 。 


男 一 个 值得 注意 的 是 Hashtable〈( 散 列表 ) ， 它 是 另 一 个 重要 的 标准 类 。 
该 类 没有 采用 任何 final 方 法 。 正 如 我 们 在 本 书 其 他 地 方 提 到 的 那样 ， 显 
然 一 些 类 的 设计 人 员 与 其 他 设计 人 员 有 着 全 然 不 同 的 素质 (注意 比较 

Hashtable 极 短 的 方法 名 与 Vecor 的 方法 名 ) 。 对 类 库 的 用 户 来 说 ， 这 显 
然 是 不 应 该 如 此 轻易 就 能 看 出 的 。 一 个 产品 的 设计 变 得 不 一 致 后 ， 会 加 
A 的 工作 量 。 这 也 从 另 一 个 侧面 强调 了 代码 设计 与 检查 时 需要 很 强 
IAE Ùo 














6.9 初始 化 和 类 装载 


在 许多 传统 语言 里 ， 程 序 都 是 作为 局 动 过 程 的 一 部 分 一 次 性 载 入 的 。 随 
后 进行 的 是 初始 化 ， 再 是 正式 执行 程序 。 在 这 些 语言 中 ， 必 须 对 初始 化 
过 程 进行 慎重 的 控制 ， 保 证 static 数 据 的 初始 化 不 会 带 来 碎 烦 。 比 如 在 一 
个 static 数 据 获得 初始 化 之 前 ， 束 有 为 一 个 static 数 据 布 望 它 是 一 个 有 效 
值 ， 那 么 在 C++ 中 就 会 造成 问题 。 


Java 则 没有 这 样 的 问题 ， 因 为 它 采 用 了 不 同 的 装载 方法 。 由 于 Java 中 的 
一 切 东西 都 是 对 象 ， 所 以 许多 活动 变 得 更 加 简单 ， 这 个 问题 便 是 其 中 的 
一 例 。 正 如 下 一 章 会 讲 到 的 那样 ， 每 个 对 象 的 代码 都 存在 于 独立 的 文件 
中 。 除 非 真 的 需要 代码 ， 人 否则 那个 文件 是 不 会 载 入 的 。 通 音 ， 我 们 可 认 
为 除非 那个 类 的 一 个 对 象 构造 完毕 ， 人 否则 代码 不 会 真 的 载 入 。 由 于 static 
人 


首次 使 用 的 地 方 也 是 static 初 始 化 发 生 的 地 方 。 装 载 的 时 候 ， 所 有 static 
对 象 和 static 代 码 块 都 会 按照 本 来 的 顺序 初始 化 〈“ 亦 即 它 们 在 类 定义 代码 
里 写 入 的 顺序 ) 。 当 然 ，static 数 据 只 会 初始 化 一 次 。 

6.9.1 继承 初始 化 


我 们 有 必要 对 整个 初始 化 过 程 有 所 认识 ， 其 中 包括 继承 ， 对 这 个 过 程 中 
发 生 的 事情 有 一 个 整体 性 的 概念 。 请 观察 下 述 代 码 : 


//: Beetle.java 




















// The full process of initialization. 
class Insect { 

int i = 9; 

int j; 

Insect() { 


prt("1 = " + i + To j = " + j); 


static int x1 = 
prt("static Insect.x1 initialized"); 
static int prt(String s) { 
System.out.println(s); 


return 47; 


i 


public class Beetle extends Insect { 


int k = prt("Beetle.k initialized"); 


Beetle() { 
prt("k =" +k); 
pre("j =" + j); 
} 


static int x2 = 
prt("static Beetle.x2 initialized"); 
static int prt(String s) { 
System.out.println(s); 


return 63; 


} 


public static void main(String[] args) { 


prt("Beetle constructor"); 


Beetle b = new Beetle(); 


J 


FA ss 


该 程序 的 输出 如 下 : 


static Insect.x initialized 


static Beetle.x initialized 
Beetle constructor 
i=9, j =0 


Beetle.k initialized 





对 Beetle 运 行 Java 时 ， 发 生 的 第 一 件 事情 是 装载 程序 到 外 面 找到 那个 
类 。 在 装载 过 程 中 ， 闭 载 程序 注意 它 有 一 个 基础 类 《 即 extends 天 键 字 要 
表达 的 意思 ) ， 所 以 随 之 将 其 载 入 。 无 论 是 否 准备 生成 那个 基础 类 的 一 
个 对 象 ， 这 个 过 程 都 会 发 生 ( 请 试 着 将 对 象 的 创建 代码 当 作 注释 标注 出 
K, HOREK) 。 
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推 。 接 下 来 ， 会 在 根基 础 类 《此 时 是 Insect) 执行 static 初 始 化 ， 再 在 下 
一 个 衍生 类 执行 ， 以 此 类 推 。 保 证 这 个 顺序 是 非常 关键 的 ， 因 为 衍生 类 
的 初始 化 可 能 要 依赖 于 对 基础 类 成 员 的 正确 初始 化 。 


此 时 ， 必 要 的 类 已 全 部 装载 完毕 ， 所 以 能 够 创建 对 象 。 首 和 完 ， 这 个 对 象 
中 的 所 有 基本 数据 类 型 部 会 设 成 它们 的 默认 值 ， 而 将 对 象 句柄 设 为 
null。 随 后 会 调用 基础 类 构建 器 。 在 这 种 情况 下 ， 调 用 是 上 自动 进行 的 。 
但 也 完全 可 以 用 super 来 自行 指定 构建 右 调 用 〈 束 象 在 Beetle() 构 建 费 中 
的 第 一 个 操作 一 样 ) 。 基 础 类 的 构建 床 用 与 衍生 类 构建 天 完全 相同 的 处 














理 过 程 。 基 础 顺 构 建 句 完成 以 后 ， 实 例 变量 会 按 本 来 的 顺序 得 以 初始 
化 。 最 后 ， 执 行 构建 器 剩余 的 主体 部 分 。 


6.10 总 结 


无 论 继承 还 是 合成 ， 我 们 都 可 以 在 现 有 类 型 的 基础 上 创建 一 个 新 类 型 。 
但 在 典型 情况 下 ， 我 们 通过 合成 来 实现 现 有 类型 的 “再 生 ? 或 “重复 使 
用 ”， 将 其 作为 新 类 型 基础 实施 过 程 的 一 部 分 使 用 。 但 如 果 想 实现 接口 
的 “再 生 ”， 就 应 使 用 继承 。 由 于 衍生 或 派生 出 来 的 类 拥有 基础 类 的 接 
口 ， 所 以 能 够 将 其 "上 漳 造 型 "为 基础 类 。 对 于 下 一 章 要 讲述 的 多 形 性 问 
题 ， 这 一 点 是 至 关 重 要 的 。 


尽管 继承 在 面 问 对 象 的 程序 设计 中 得 到 了 特别 的 强调 ， 但 在 实际 月 动 一 
个 设计 时 ， 最 好 还 是 先 考 虑 采用 合成 技术 。 只 有 在 特别 必要 的 时 候 ， 才 
应 考虑 采用 继承 技术 《下 一 章 还 会 讲 到 这 个 问题 ) 。 合 成 显得 更 加 灵 
活 。 但 是 ， 通 过 对 上 自己 的 成 员 类 型 应 用 一 些 继承 技巧 ， 可 在 运行 期 准确 
改变 那些 成 员 对 象 的 类 型 ， 由 此 可 改变 它们 的 行为 。 


尽管 对 于 快速 项 目 开 发 来 说 ， 通 过 合成 和 继承 实现 的 代码 再 生 具 有 很 大 
的 帮助 作用 。 但 在 允许 其 他 程序 员 完 全 依赖 它 之 前 ， 一 般 都 希望 能 重新 
设计 目 己 的 类 结构 。 我 们 理想 的 类 结构 应 该 是 每 个 类 都 有 目 己 特定 的 用 
途 。 它 们 不 能 过 大 如 集成 的 功能 太 多 ， 则 很 难 实 现 它 的 再 生 ) ， 也 不 
能 过 小 《造成 不 能 由 目 己 使 用 ， 或 者 不 能 增添 新 功能 ) 。 最 终 实现 的 类 
应 该 能 够 方便 地 再 生 。 























6.11 练习 


(1) 用 默认 构建 器 《〈 空 自 变量 列表 ) 创建 两 个 类 : A 和 B， 令 它们 自己 声 
明 上 自己。 从 A 继承 一 个 名 为 C 的 新 类 ， 并 在 C 内 创建 一 个 成 员 B。 不 要 为 
C 创 建 一 个 构建 器 。 创 建 类 C 的 一 个 对 象 ， 并 观察 结果 。 


(2) ”修改 练习 1， 使 A 和 B 都 有 含有 目 变 量 的 构建 况 ， 则 不 是 采用 默认 构 
建 器 。 为 C 写 一 个 构建 器 ， 并 在 C 的 构建 器 中 执行 所 有 初始 化 工作 。 


(3) 使 用 文件 Cartoon.java， 将 Cartoon 类 的 构建 器 代码 变 成 注释 内 容 标注 
出 去 。 解 释 会 发 生 什 么 事情 。 


(4) 使 用 文件 Chess.java， 将 Chess 类 的 构建 器 代码 作为 注释 标注 出 去 。 同 
样 解释 会 发 生 什 么 。 











第 7 章 多 形 性 


“对 于 面向 对 象 的 程序 设计 语言 ， 多 型 性 是 第 三 种 最 基本 的 特征 《前 两 
种 是 数据 抽象 和 继承 。” 


“多 形 性 ”(Polymorphism ) 从 另 一 个 角度 将 接口 从 具体 的 实施 细节 中 分 
离 出 来 ， 亦 即 实现 了 “是 什么 ”与 “怎样 做 ?两 个 模块 的 分 离 。 利 用 多 形 性 
的 概念 ， 代 码 的 组 织 以 及 可 读 性 均 能 获得 改善 。 此 外 ， 还 能 创建 “易于 
扩展 ”的 程序 。 无 论 在 项 目的 创建 过 程 中 ， 还 是 在 需要 加 入 新 特性 的 时 
修 ， 它 们 都 可 以 方便 地 “成 长 ”。 


通过 合并 各 种 特征 与 行为 ， 封 装 技术 可 创建 出 新 的 数据 类 型 。 通 过 对 具 
体 实 施 细节 的 隐藏 ， 可 将 接口 与 实施 细节 分 离 ， 使 所 有 细节 成 

为 “private”( 私 有 )，。 这 种 组 织 方 式 使 那些 有 程序 化 编程 背景 人 感觉 颇 
为 舒适 。 但 多 形 性 却 涉及 对 “类 型 ”的 分 解 。 通 过 上 一 章 的 学 习 ， 大 家 已 
知道 通过 继承 可 将 一 个 对 象 当 作 它 自己 的 类 型 或 者 它 自 己 的 基础 类 型 对 
待 。 这 种 能 力 是 十 分 重要 的 ， 因 为 多 个 类 型 (从 相同 的 基础 类 型 中 衍生 
出 来 ) 可 被 当 作 同一 种 类 型 对 符 。 而 且 只 需 一 段 代 码 ， 即 可 对 所 有 不 同 
的 类 型 进行 同样 的 处 理 。 利 用 具有 多 形 性 的 方法 调用 ， 一 种 类 型 可 将 自 
己 与 男 一 种 相似 的 类 型 区 分 开 ， 只 要 它们 都 是 从 相同 的 基础 类 型 中 衍生 
出 来 的 。 这 种 区 分 是 通过 各 种 方法 在 行为 上 的 差异 实现 的 ， 可 通过 基础 
类 实现 对 那些 方法 的 调用 。 


在 这 一 章 中 ， 大 家 要 由 浅 入 深 地 学 习 有 关 多 形 性 的 问题 〈 也 叫 作 动态 绑 
定 、 推 迟 绑 定 或 者 运行 期 绑 定 ) 。 同 时 举 一 些 简单 的 例子 ， 其 中 所 有 无 
关 的 部 分 部 已 剥 除 ， 只 保留 与 多 形 性 有 关 的 代码 。 
































7.1 上 漳 造 型 


在 第 6 音 ， 大 家 已 知道 可 将 一 个 对 象 作 为 它 目 己 的 类 型 使 用 ， 或 者 作为 

它 的 基础 类 型 的 一 个 对 象 使 用 。 取 得 一 个 对 象 句柄 ， 并 将 其 作为 基础 类 

本 1 的 行为 就 叫 作 “* 上 漳 造 型 一 一 因为 继承 树 的 画 法 是 基础 类 位 
最 上 方 。 


但 这 样 做 也 会 遇 到 一 个 问题 ， 如 下 例 所 示 《〈 知 执行 这 个 程序 遇 到 麻烦 ， 
请 参考 第 3 章 的 3.1.2 小 节 “ 赋 值 >) : 


//: Music.java 





// Inheritance & upcasting 
package c07; 
class Note { 
private int value; 
private Note(int val) { value = val; } 
public static final Note 
middleC = new Note(0), 
cSharp = new Note(1), 
cFlat = new Note(2); 
} // Etc. 
class Instrument { 
public void play(Note n) { 


System.out.printin("Instrument.play()"); 


j 


// Wind objects are instruments 
// because they have the same interface: 
class Wind extends Instrument { 
// Redefine interface method: 
public void play(Note n) { 


System.out.println("Wind.play()"); 


} 
public class Music { 
public static void tune(Instrument i) { 
TT sie 
i.play(Note.middleC); 
} 
public static void main(String[] args) { 
Wind flute = new Wind(); 
tune(flute); // Upcasting 
} 
} ///:~ 


其 中 ， 方 法 Music.tune0 接 收 一 个 mstrument 句 柄 ， 同 时 也 接收 从 
Instrument 衍 生出 来 的 所 有 东西 。 当 一 个 Wind 句 柄 传递 给 tuneO 的 时 候 ， 
天 会 出 现 这 种 情况 。 此 时 没有 造型 的 必要 。 这 样 做 是 可 以 接受 的 ; 
Instrument 里 的 接口 必须 存在 于 Wind 中 ， 因 为 Wind 是 从 Instrument 里 继承 
得 到 的 。 从 Wind 癌 Instrument 的 上 漳 造 型 可 能 “缩小 ?那个 接口 ， 但 不 可 





能 把 它 变 得 比 Pnstrument 的 完整 接口 还 要 小 。 
7.1.1 为 什么 要 上 渊 造型 


这 个 程序 看 起 来 也 许 显得 有 些 奇 怪 。 为 什么 所 有 人 都 应 该 有 意 忘记 一 个 
对 象 的 类 型 呢 ? 进 行 上 济 造 型 时 ， 束 可 能 产生 这 方面 的 疑惑 。 而 且 如 果 
让 tuneO 简 单 地 取得 一 个 wind 句 柄 ， 将 其 作为 自己 的 自 变 量 使 用 ， 似 乎 
会 更 加 简单 、 直 观 得 多 。 但 要 注意 : 假如 那样 做 ， 就 需 为 系统 内 
instrument 的 每 种 类 型 写 一 个 全 新 的 tne0。 假 设 按照 前 面 的 推论 ， 加 入 
Stringed (5475) MBrass (HE) XPP Instrument (FR AS) : 


























//: Music2.java 


// Overloading instead of upcasting 
class Note2 { 
private int value; 
private Note2(int val) { value = val; } 
public static final Note2 
middleC = new Note2(0), 
cSharp = new Note2(1), 
cFlat = new Note2(2); 
} // Etc. 
class Instrument2 { 
public void play(Note2 n) { 


System.out.printin("Instrument2.play()"); 


} 


class Wind2 extends Instrument2 { 


public void play(Note2 n) { 


System.out.println("Wind2.play()"); 


J 


class Stringed2 extends Instrument2 { 
public void play(Note2 n) { 


System.out.println("Stringed2.play()"); 


} 


class Brass2 extends Instrument2 { 
public void play(Note2 n) { 


System.out.println("Brass2.play()"); 


} 


public class Music2 { 

public static void tune(Wind2 i) { 
i.play(Note2.middleC); 

} 

public static void tune(Stringed2 i) { 
i.play(Note2.middleC); 

} 

public static void tune(Brass2 i) { 


1.play(Note2.middleC) ; 


} 

public static void main(String[] args) { 
Wind2 flute = new Wind2(); 
Stringed2 violin = new Stringed2(); 
Brass2 frenchHorn = new Brass2(); 
tune(flute); // No upcasting 
tune(violin); 
tune(frenchHorn); 

} 


} ///:~ 


这 样 做 当然 行 得 通 ， 但 却 存 在 一 个 极 大 的 弊端 : 必须 为 每 种 新 增 的 
Instrument2 类 编写 与 类 紧密 相关 的 方法 。 这 意味 着 第 一 次 束 要 求 多 得 多 
的 编程 量 。 以 后 ， 假 如 想 添 加 一 个 象 tune() 那 样 的 新 方法 或 者 为 
Instrument 添 加 一 个 新 类 型 ， 仍 然 需 要 进行 大 量 编码 工作 。 此 外 ， 即 使 
态 记 对 自己 的 某 个 方法 进行 过 载 设置 ， 编 译 器 也 不 会 提示 任何 错误 。 这 
样 一 来 ， 类 型 的 整个 操作 过 程 就 显得 极 难 管 理 ， 有 失控 的 危险 。 


但 假如 只 写 一 个 方法 ， 将 基础 类 作为 日 变量 或 参数 使 用 ， 而 不 是 使 用 那 
些 特定 的 衍生 类 ， 关 不 是 会 简单 得 多 ? 也 就 是 说 ， 如 果 我 们 能 不 顾 衍生 
和 
dE 


这 正 是 “多 形 性 ”大 显 映 手 的 地 方 。 然 而 ， 大 多 数 程序 员 (特别 是 有 程序 
化 编程 背景 的 ) 对 于 多 形 性 的 工作 原理 仍然 显得 有 些 生 琉 。 




















7.2 深入 理解 


对 于 Mnusic.java 的 困难 性 ， 可 通过 运行 程序 加 以 体会 。 输 出 是 
Wind.play0。 这 当然 是 我 们 希望 的 输出 ， 但 它 看 起 来 似乎 并 不 愿 按 我 们 
的 希望 行事 。 请 观察 一 下 tune0 方 法 : 








public static void tune(Instrument i) { 
ieee 

i.play(Note.middleC); 

} 


它 接收 Instrument 人 句柄 。 所 以 在 这 种 情况 下 ， 编 译 器 怎样 才能 知道 
Instrument 人 句柄 指 回 的 是 一 个 wind， 而 不 是 一 个 Brass 或 Stringed 呢 ? 编译 
器 无 从 得 知 。 为 了 深入 了 理解 这 个 问题 ， 我 们 有 必要 探讨 一 下 “ 绑 定 ”这 


个 主题 。 
7.2.1 方法 调用 的 绑 定 


将 一 个 方法 调用 同一 个 方法 主体 连接 到 一 起 束 称 为 “ 绑 定 ”(Binding)。 
右 在 程序 运行 以 前 执行 绑 定 (由 编译 融和 链接 程序 ， 如 朵 有 的 话 )， 器 
叫 作 “早期 绑 定 ”。 大 家 以 前 或 许 从 未 听 说 过 这 个 术语 ， 因 为 它 在 任何 程 
序 化 语言 里 都 是 不 可 能 的 。C 编 译 器 只 有 一 种 方法 调用 ， 那 就 是 < 早期 绑 


上 述 程序 最 令 人 迷惑 不 解 的 地 方 全 与 早期 绑 定 有 关 ， 因 为 在 只 有 一 个 
Instrument 人 句柄 的 前 提 下 ， 编 译 器 不 知道 具体 该 调用 哪个 方法 。 


解决 的 方法 就 是 “后 期 绑 定 ”， 它 意味 着 绑 定 在 运行 期 间 进 行 ， 以 对 象 的 
类 型 为 基础 。 后 期 绑 定 也 叫 作 "动态 绑 定 ?或 “运行 期 绑 定 ”。 知 一 种 语言 
实现 了 后 期 绑 定 ， 同 时 必须 提供 一 些 机 制 ， 可 在 运行 期 间 判 断 对 象 的 类 
型 ， 并 分 别 调用 适当 的 方法 。 也 就 是 说 ， 编 译 圳 此 时 依然 不 知道 对 象 的 
类 型 ， 但 方法 调用 机 制 能 自己 去 调查 ， 找 到 正确 的 方法 主体 。 不 同 的 语 
言 对 后 期 绑 定 的 实现 方法 是 有 所 区 别 的 。 但 我 们 至 少 可 以 这 样 认 为 : 它 
们 都 要 在 对 象 中 安插 茶 些 特殊 类 型 的 信息 。 


























Java 中 绑 定 的 所 有 方法 都 采用 后 期 绑 定 技术 ， 除 非 一 个 方法 已 被 声明 成 
Ra XERE RINE E A UR E E FR LIE AT Ja HRE — E E AR 


为 什么 要 把 一 个 方法 声明 成 final 呢 ? 正如 上 一 重 指 出 的 那样 ， 它 能 防止 
其 他 人 履 兰 那个 方法 。 但 也 许 更 重要 的 一 点 是 ， 它 可 有 效 地 “关闭 ?动态 
绑 定 ， 或 者 告诉 编译 器 不 需要 进行 动态 绑 定 。 这 样 一 来 ， 编 译 器 就 可 为 
final 方 法 调用 生成 效率 更 高 的 代码 。 


7.2.2 产生 正确 的 行为 


知道 Java 里 绑 定 的 所 有 方法 都 通过 后 期 绑 定 具有 多 形 性 以 后 ， 束 可 以 相 
应 地 编写 目 己 的 代码 ， 令 其 与 基础 类 沟通 。 此 时 ， 所 有 的 衍生 类 都 保证 
能 用 相同 的 代码 正常 地 工作 。 或 者 换 用 男 一 种 方法 ， 我 们 可 以 “将 一 条 
消息 发 给 一 个 对 象 ， 让 对 象 上 自行 判断 要 做 什么 事情 。” 


在 面 癌 对象 的 程序 设计 中 ， 有 一 个 经 典 的 “形状 ”例子 。 由 于 它 很 容易 用 
可 视 化 的 形式 表现 出 来 ， 所 以 经 常 都 用 它 说 明 问 题 。 但 很 不 池 的 是 ， 它 
A erp en ines 这 种 认识 当然 是 错 
误 的 。 

形状 例子 有 一 个 基础 类 ， 名 为 Shape; 另外 还 有 大 量 衍 生 类 型 : 

Circle〈 圆 形 ) , Square (方形 ) , Triangle (SAW) 等 等 。 大 家 之 所 


以 喜欢 这 个 例子 ， 因 为 很 容易 理解 <“ 圆 属于 形状 的 一 种 类 型 等 概念 。 下 
面 这 幅 继 承 图 向 我 们 展示 了 它们 的 关系 : 


draw) 
Ky 




















Cast "up" the 
inheritance A 
diagram 


r: 一 :一 :一 一 :一 :一 .一 :一 






Circle draw() 
Handle erase() 


上 漳 造 型 可 用 下 面 这 个 语句 简单 地 表现 出 来 : 





Shape s = new Circle(); 

在 这 里 ， 我 们 创建 了 Circle 对 象 ， 并 将 结果 句柄 立即 赋 给 一 个 Shape。 这 
表面 看 起 来 似乎 属于 错误 操作 (将 一 种 类 型 分 配给 男 一 个 ) ， 但 实际 是 
完全 可 行 的 因为 按照 继承 关系 ，Circle 属 于 Shape 的 一 种 。 因 此 编译 
器 认可 上 述 语句 ， 不 会 回 我 们 提示 一 条 出 错 消息 。 


当 我 们 调用 其 中 一 个 基础 类 方法 时 《已 在 衍生 类 里 复 盖 ) : 








s.draw(); 


同样 地 ， 大 家 也 许 认 为 会 调用 Shape 的 draw(0， 因 为 这 毕竟 是 一 个 Shape 
人 句柄。 那么 编译 器 怎样 才能 知道 该 做 其 他 任何 事情 呢 ? 但 此 时 实际 调用 
的 是 Circle.draw()， 因 为 后 期 绑 定 已 经 介入 (多 形 性 ) 。 

下 面 这 个 例子 从 一 个 稍微 不 同 的 角度 说 明了 问题 : 


//: Shapes.java 








// Polymorphism in Java 
class Shape { 
void draw() {} 
void erase() {} 
} 
class Circle extends Shape { 
void draw() { 
System.out.println("Circle.draw()"); 
} 
void erase() { 


System.out.printin("Circle.erase()"); 


} 


class Square extends Shape { 
void draw() { 
System.out.println("Square.draw()"); 
} 
void erase() { 


System.out.println("Square.erase()"); 


} 


class Triangle extends Shape { 
void draw() { 
System.out.printin("Triangle.draw()"); 
} 
void erase() { 


System.out.println("Triangle.erase()"); 


} 


public class Shapes { 
public static Shape randShape() { 
switch((int)(Math.random() * 3)) { 
default: // To quiet the compiler 


case 0: return new Circle(); 


case 1: return new Square(); 


case 2: return new Triangle(); 


} 
public static void main(String[] args) { 
Shape[] s = new Shape[9]; 
// Fill up the array with shapes: 
for(int i = 0; i < s.length; i++) 
s[i] = randShape(); 
// Make polymorphic method calls: 
for(int i = 0; i < s.length; i++) 
s[i].draw(); 
} 
XA 


针对 从 Shape 衍 生出 来 的 所 有 东西 ，Shape 建 立 了 一 个 通用 接口 一 一 也 就 
Æw, MA JLD 形状 都 可 以 描绘 和 删除 。 衍 生 类 履 盖 了 这 些 定义 ， 
为 每 种 特殊 类 型 的 几何 形状 都 提供 了 独一无二 的 行为 。 


在 主 类 Shapes 里 ， 包 含 了 一 个 static 方 法 ， 名 为 randShape0。 它 的 作用 是 
在 每 次 调用 它 时 为 某 个 随机 选择 的 Shape 对 象 生成 一 个 句柄 。 请 注意 上 
溯 造 型 是 在 每 个 returmn 语 句 里 发 生 的 。 这 个 语句 取得 指 问 一 个 Circle， 
Square 或 者 Triangle 的 句柄 ， 并 将 其 作为 返回 类 型 Shape 发 给 方法 。 所 以 
无 论 什 么 时 候 调 用 这 个 方法 ， 就 绝对 没 机 会 了 解 它 的 具体 类 型 到 底 是 什 
么 ， 因 为 肯定 会 获得 一 个 单纯 的 Shape 句 柄 。 


main() 包 含 了 Shape 人 句柄 的 一 个 数组 ， 其 中 的 数据 通过 对 randShape() 的 调 
用 填 入 。 在 这 个 时 候 ， 我 们 知道 自己 拥有 Shape， 但 不 知 除 此 之 外 任何 


具体 的 情况 《编译 器 同样 不 知 ) 。 然 而 ， 当 我 们 在 这 个 数组 里 步 进 ， 并 
为 每 个 元 了 素 调用 draw0O 的 时 候 ， 与 各 类 型 有 关 的 正确 行为 会 魔术 般 地 发 
生 ， 瓯 朝 下 面 这 个 输出 示例 展示 的 那样 : 


Circle.draw() 





Triangle.draw() 
Circle.draw() 
Circle.draw() 
Circle.draw() 
Square.draw( ) 
Triangle.draw() 
Square.draw( ) 


Square.draw( ) 





当然 ， 由 于 几何 形状 是 每 次 随机 选择 的 ， 所 以 每 次 运行 都 可 能 有 不 同 的 
结 末 。 之 所 以 要 突出 形状 的 随机 选择 ， 是 为 了 让 大 家 深刻 体会 这 一 点 : 
为 了 在 编译 的 时 候 发 出 正确 的 调用 ， 编 译 器 毋 需 获得 任何 特殊 的 情报 。 
对 drawO 的 所 有 调用 都 是 通过 动态 绑 定 进行 的 。 


7.2.3 扩展 性 


现在 ， 让 我 们 仍然 返回 乐器 Cnstrument) 示例 。 由 于 存在 多 形 性 ， 所 
以 可 根据 目 己 的 需要 同系 统 里 加 入 任意 多 的 新 类 型 ， 同 时 考 需 更 改 
true() 方 法 。 在 一 个 设计 良好 的 OOP 程 序 中 ， 我 们 的 大 多 数 或 者 所 有 方 
法 都 会 遵从 tune0 的 模型 ， 而 且 只 与 基础 类 接口 通信 。 我 们 说 这 样 的 程 
序 具有 “扩展 性 *”， 因 为 可 以 从 通用 的 基础 类 继承 新 的 数据 类 型 ， 从 而 新 
添 一 些 功能 。 如 果 是 为 了 适应 新 类 的 要 求 ， 那 么 对 基础 类 接口 进行 操纵 
的 方法 根本 不 需要 改变 ， 


对 于 乐 费 例子 ， 假 设 我 们 在 基础 类 里 加 入 更 多 的 方法 ， 以 及 一 系列 新 
类 ， 那 么 会 出 现 什么 情况 呢 ? 下 面 是 示意 图 : 























void play) 
String whatQ) 
void adjust() 









void play) 


String whatt) 
void adjust(} 


Percussion 


void play) 
String whatQ) 
void adjust() 


Stringed 

















void play) 
String what) 
void adjust(} 







void play) 
String what) 


void play) 
void adjust() 


所 有 这 些 新 类 都 能 与 老 类 一 tune RRE, De tune EE 
整 。 即 使 tuneO 位 于 一 个 独立 的 文件 里 ， 而 将 新 方法 添加 到 Instrument 的 
接口 ，tune0 也 能 正确 地 工作 ， 不 需要 重新 编译 。 下 面 这 个 程序 是 对 上 
述 示意 图 的 具体 实现 : 


//: Music3.java 











// An extensible program 
import java.util.*; 
class Instrument3 { 
public void play() { 
System.out.println("Instrument3.play()"); 
} 


public String what() { 


return "Instrument3"; 


i; 


public void adjust() {} 
} 
class Wind3 extends Instrument3 { 
public void play() { 
System.out.println("Wind3.play()"); 
} 
public String what() { return "Wind3"; } 
public void adjust() {} 
} 
class Percussion3 extends Instrument3 { 
public void play() { 
System.out.println("Percussion3.play()"); 
} 
public String what() { return "Percussion3"; } 
public void adjust() {} 
} 
class Stringed3 extends Instrument3 { 
public void play() { 
System.out.println("Stringed3.play()"); 


} 


public String what() { return "Stringed3"; } 


public void adjust() {} 
} 
class Brass3 extends Wind3 { 
public void play() { 
System.out.println("Brass3.play()"); 
J 
public void adjust() { 


System.out.println("Brass3.adjust()"); 


} 


Class Woodwind3 extends Wind3 { 

public void play() { 

System.out.println("Woodwind3.play()"); 

} 

public String what() { return "Woodwind3"; } 
} 
public class Music3 { 

// Doesn't care about type, so new types 
// added to the system still work right: 
static void tune(Instrument3 1) { 


LI oa 


i.play(); 


static void tuneAll(Instrument3[] e) { 

for(int i = 0; i < e.length; i++) 
tune(e[i]); 

} 

public static void main(String[] args) { 
Instrument3[] orchestra = new Instrument3[5]; 
int i = 0; 
// Upcasting during addition to the array: 
orchestra[it++] = new Wind3(); 
orchestra[it+t+] = new Percussion3(); 
orchestra[it+t+] = new Stringed3(); 
orchestra[it++] = new Brass3(); 
orchestra[i++] = new Woodwind3(); 
tuneAll(orchestra) ; 

} 

} ///:~ 


新 方法 是 whatO 和 adjust()。 前 者 返回 一 个 String 人 句柄 ， 同 时 返回 对 那个 类 
的 说 明 ;， 后 者 使 我 们 能 对 每 种 乐器 进行 调整 。 


在 main() 中 ， 当 我 们 将 某 样 东西 置 入 Instrument3 数 组 时 ， 束 会 自动 上 济 


造型 到 Instrument3。 


可 以 看 到 ， 在 围绕 tune(0) 方 法 的 其 他 所 有 代码 都 发 生变 化 的 同时 ，tune0) 
方法 却 丝 坚 不 受 它们 的 影响 ， 依 然 故我 地 正常 工作 。 这 正 是 利用 多 形 性 
希望 达到 的 目标 。 我 们 对 代码 进行 修改 后 ， 不 会 对 程序 中 不 应 受到 影 啊 
的 部 分 造成 影响 。 此 外 ， 我 们 认为 多 形 性 是 一 种 至 关 重 要 的 技术 ， 它 允 





许 程序 员 “ 将 发 生 改 变 的 东西 同 没有 发 生 改 变 的 东西 区 分 开 ”。 


7.3 4 i WAX 


现在 让 我 们 用 不 同 的 眼光 来 看 看 本 章 的 头 一 个 例子 。 在 下 面 这 个 程序 
中 ， 方 法 play0O 的 接口 会 在 被 上 覆盖 的 过 程 中 发 生变 化 。 这 意味 独 我 们 实 
际 并 没有 “ 窗 盖 ”方法 ， 而 是 使 其 “过 载 *。 编 译 器 允许 我 们 对 方法 进行 过 
载 处 理 ， 使 其 不 报告 出 错 。 但 这 种 行为 可 能 并 不 是 我 们 所 希望 的 。 下 面 
是 这 个 例子 : 


//: WindError.java 








// Accidentally changing the interface 
class Notex { 
public static final int 
MIDDLE_C = ©, C_SHARP = 1, C_FLAT = 2; 
} 
class InstrumentX { 
public void play(int NoteX) { 


System.out.printin("InstrumentX.play()"); 


} 


class WindX extends InstrumentxX { 
// OOPS! Changes the method interface: 
public void play(Notex n) { 


System.out.println("WindxX.play(Notex n)"); 


public class WindError { 
public static void tune(InstrumentxX i) { 
fe as 
1.play(NotexX.MIDDLE_C); 
} 
public static void main(String[] args) { 
WindxX flute = new WindX(); 
tune(flute); // Not the desired behavior! 
} 
二 





这 里 还 癌 大 家 引入 了 另 一 个 易于 混 消 的 概念 。 在 InstrumentX 中 ，Pplay0) 
方法 采用 了 一 个 int (整数 ) 数值 ， 它 的 标识 符 是 NoteX。 也 就 是 说 ， 即 
使 NoteX 是 一 个 类 名 ， 也 可 以 把 它 作 为 一 个 标识 符 使 用 ， 编 译 器 不 会 报 
告 出错 。 但 在 Windx 中 ，play0 末 用 一 个 NoteX 人 句柄 ， 它 有 一 个 标识 符 
n。 即 便 我 们 使 用 “play(NoteX NoteX)”， 编 译 器 也 不 会 报告 错误 。 这 样 

K, ARERR ETET RA at mplay) kp, 但 对 方法 的 类 型 定 
义 却 稍微 有 些 不 确切 。 然 而 ， 编 译 器 此 时 假定 的 是 程序 员 有 意 进 行 “ 过 
载 ?， 而 非 “ 履 盖 ”。 请 仔细 体会 这 两 个 术语 的 区 别 。 “ER? RET I 
东西 在 不 同 的 地 方 具 有 多 种 含义 ; 而 “ 履 盖 ?是 指 它 随时 随地 都 只 有 一 种 
含义 ， 只 是 原先 的 含义 完全 被 后 来 的 含义 取代 了 。 请 注意 如 果 遵 守 标 准 
的 Java 命 名 规范 ， 自 变 量 标识 符 就 应 该 是 noteX， 这 样 可 把 它 与 类 名 区 
分 开 。 


在 tune 中 , “InstrumentX ”六 会 发 出 play0) 消 息 ， 同 时 将 某 个 NoteX 成 员 作 
为 自 变 量 使 用 CMIDDLE_C) 。 由 于 NoteX 包 含 了 int 定 义 ， 过 载 的 play(O) 
方法 的 int 版 本 会 得 到 调用 。 同 时 由 于 它 尚 未 被 “ 绑 闸 >， 所 以 会 使 用 基础 


类 版 本 


输出 是 : 
































InstrumentX.play() 





7.4 抽象 类 和 方法 


在 我 们 所 有 乐 问 〈Instrument) 例子 中 ， 基 础 类 Instrument 内 的 方法 都 肯 
定 是 “ 伪 ” 方 法 。 若 去 调用 这 些 方法 ， 束 会 出 现 错误 。 那 是 由 于 
Instrument 的 意图 是 为 从 它 衍 生出 去 的 所 有 类 都 创建 一 个 通用 接口 。 


之 所 以 要 建立 这 个 通用 接口 ， 唯 一 的 原因 就 是 它 能 为 不 同 的 子 类 型 作出 
不 同 的 表示 。 它 为 我 们 建立 了 一 种 基本 形式 ， 使 我 们 能 定义 在 所 有 衍生 
类 里 “通用 ”的 一 些 东西 。 为 前 述 这 个 观念 ， 另 一 个 方法 是 把 Instrument 
称 为 “抽象 基础 类 ”简称 “抽象 类 ”) 。 若 想 通 过 该 通用 接口 处 理 一 系列 
类 ， 就 需要 创建 一 个 抽象 类 。 对 所 有 与 基础 类 声明 的 签名 相符 的 衍生 类 
方法 ， 都 可 以 通过 动态 绑 定 机 制 进行 调用 《〈 然 而， 正如 上 一 节 指 出 的 那 
样 ， 如 果 方 法 名 与 基础 类 相同 ， 但 自 变 量 或 参数 不 同 ， 就 会 出 现 过 载 现 
象 ， 那 或 许 并 非 我 们 所 愿意 的 )。 


如 果 有 一 个 象 Instrument 那 样 的 抽象 类 ， 那 个 类 的 对 象 几 乎 肯定 没有 什 
么 意义 。 换 言 之 ，Imstrument 的 作用 仅仅 是 表达 接口 ， 而 不 是 表达 一 些 
具体 的 实施 细节 。 所 以 创建 一 个 Instrument 对 象 是 没有 意义 的 ， 而 且 我 
们 通常 都 应 禁止 用 户 那 样 做 。 为 达到 这 个 目的 ， 可 令 Instruament 内 的 所 
有 方法 都 显示 出 错 消 息 。 但 这 样 做 会 延迟 信息 到 运行 期 ， 并 要 求 在 用 户 
那 一 面 进行 彻底 、 可 靠 的 测试 。 无 论 如 何 ， 最 好 的 方法 都 是 在 编译 期 间 
捕捉 到 问题 。 


针对 这 个 问题 ，Java 专 门 提 供 了 一 种 机 制 ， 名 为 “抽象 方法 ”。 它 属于 一 
种 不 完整 的 方法 ， 只 含有 一 个 声明 ， 没 有 方法 主体 。 下 面 是 抽象 方法 声 
明 时 采用 的 语法 : 


abstract void X(); 

包含 了 抽象 方法 的 一 个 类 叫 作 *“ 抽 象 类 ”。 如 果 一 个 类 里 包含 了 一 个 或 多 
个 抽象 方法 ， 类 就 必须 指定 成 abstract〈 抽 象 ) 。 和 否则 ， 编 译 器 会 向 我 们 
报告 一 条 出 错 消 县 。 

若 一 个 抽象 类 是 不 完整 的 ， 那 么 一 旦 有 人 试图 生成 那个 类 的 一 个 对 象 ， 


编译 器 又 会 采取 什么 行动 呢 ? 由 于 不 能 安全 地 为 一 个 抽象 类 创建 属于 它 
的 对 象 ， 所 以 会 从 编译 器 那里 获得 一 条 出 错 提示 。 通 过 这 种 方法 ， 编 译 





























恬 可 保证 抽象 类 有 的 “纯洁 性 ”， 我 们 不 必 担 心 会 误 用 它 。 


如 果 从 一 个 抽象 类 继承 ， 而 且 想 生成 新 类 型 的 一 个 对 象 ， 就 必须 为 基础 
类 中 的 所 有 抽象 方法 提供 方法 定义 。 如 果 不 这 样 做 〈 完 全 可 以 选择 不 
(BO ， 则 衍生 关 也 会 是 抽象 的 ， 而 且 编 译 韦 会 强迫 我 们 用 abstract 天 键 字 
标志 那个 类 的 “抽象 本质。 


即使 不 包括 任何 abstract 方 法 ， 亦 可 将 一 个 类 声明 成 “抽象 类 ”。 如 果 一 个 
类 没 必 要 拥有 任何 抽象 方法 ， 而 且 我 们 想 禁 止 那个 类 的 所 有 实例 ， 这 种 
能 力 就 会 显得 非常 有 用 。 


Instrument 类 可 很 轻松 地 转换 成 一 个 抽象 类 。 只 有 其 中 一 部 分 方法 会 变 
成 抽象 方法 ， 因 为 使 一 个 类 抽象 以 后 ， 并 不 会 强迫 我 们 将 它 的 所 有 方法 
都 同时 变 成 抽象 。 下 面 是 它 看 起 来 的 样子 : 


abstract Instrument 


abstract void play); 
String what() { /* ... */ } 
abstract void adjust); 















void play) 
String what) 
void adjust() 







Percussion 


void play) 
String what) 
void adjust) 


void play 


String what() 
void adjust(} 
















Brass 


void play) 
void adjust() 


void play) 
String what() 








FAERIT EZ RRT, FLORAL SARA RTT: 


//: Music4.java 


// Abstract classes and methods 
import java.util.*; 
abstract class Instrument4 { 
int 1; // storage allocated for each 
public abstract void play(); 
public String what() { 
return "Instrument4"; 


j 


public abstract void adjust(); 
} 
class Wind4 extends Instrument4 { 
public void play() { 
System.out.println("Wind4.play()"); 
} 
public String what() { return "Wind4"; } 
public void adjust() {} 
} 
class Percussion4 extends Instrument4 { 
public void play() { 
System.out.println("Percussion4.play()"); 
} 
public String what() { return "Percussion4"; } 


public void adjust() {} 


} 


class Stringed4 extends Instrument4 { 
public void play() { 
System.out.println("Stringed4.play()"); 
} 
public String what() { return "Stringed4"; } 
public void adjust() {} 
} 
class Brass4 extends Wind4 { 
public void play() { 
System.out.println("Brass4.play()"); 
} 
public void adjust() { 


System.out.println("Brass4.adjust()"); 


} 


class Woodwind4 extends Wind4 { 
public void play() { 
System.out.println("Woodwind4.play()"); 
} 


public String what() { return "Woodwind4"; } 


} 


public class Music4 { 


// Doesn't care about type, so new types 
// added to the system still work right: 
static void tune(Instrument4 i) { 
IE. Ss 
i.play(); 
} 
static void tuneAll(Instrument4[] e) { 
for(int i = 0; i < e.length; i++) 
tune(e[i]); 
} 
public static void main(String[] args) { 
Instrument4[] orchestra = new Instrument4[5]; 
int i = 0; 
// Upcasting during addition to the array: 
orchestra[it++] = new Wind4(); 
orchestra[it+t+] = new Percussion4(); 
orchestra[it+t+] = new Stringed4(); 
orchestra[it++] = new Brass4(); 
orchestra[it++] = new Woodwind4(); 


tuneAll(orchestra) ; 


} ///:~ 


可 以 看 出 ， 除 基础 类 以 外 ， 实 际 并 没有 进行 什么 改变 。 


创建 抽象 类 和 方法 有 时 对 我 们 非常 有 用 ， 因 为 它们 使 一 个 类 的 抽象 变 成 
明显 的 事实 ， 可 明确 告诉 用 户 和 编译 器 上 自己 打算 如 何 用 它 。 





7.5 接口 


“interface” GZO) 关键 字 使 抽象 的 概念 更 深入 了 一 层 。 我 们 可 将 其 想 
象 为 一 个 “ 纯 ” 抽 象 类 。 它 允许 创建 者 规定 一 个 类 的 基本 形式 方法 名 、 
目 变 量 列表 以 及 返回 类 型 ， 但 不 规定 方法 主体 。 接 口 也 包含 了 基本 数据 
类 型 的 数据 成 员 ， 但 它们 都 默认 为 static 和 final。 接 口 只 提供 一 种 形式 ， 
并 不 提供 实施 的 细节 。 


接口 这 样 描述 目 己 :“ 对 于 实现 我 的 所 有 类 ， 看 起 来 都 应 该 象 我 现在 这 
个 样子 ”。 因 此 ， 采 用 了 一 个 特定 接口 的 所 有 代码 都 知道 对 于 那个 接口 
可 能 会 调用 什么 方法 。 这 便 是 接口 的 全 部 含义 。 所 以 我 们 常 把 接口 用 于 
建立 类 和 类 之 间 的 一 个 “协议 ”。 有 些 面 问 对 象 的 程序 设计 语言 采用 了 一 
个 名 为 “protocol”( 协 议 ) 的 关键 子 ， 它 做 的 便 是 与 接口 相同 的 事情 。 


为 创建 一 个 接口 ， 请 使 用 interface 关 键 字 ， 而 不 要 用 class。 与 类 相似 ， 
我 们 可 在 interface 关 键 字 的 前 面 增加 一 个 public 关 键 字 (但 只 有 接口 定义 
于 同名 的 一 个 文件 内 ) ; 或 者 将 其 省 略 ， 营 造 一 种 “友好 的 ”状态 。 


为 了 生成 与 一 个 特定 的 接口 〈 或 一 组 接口 ) 相符 的 类 ， 要 使 用 
implements (SCHL) 关键 字 。 我 们 要 表达 的 意思 是 “接口 看 起 来 就 象 那个 
样子 ， 这 儿 是 它 具 体 的 工作 细节 ”。 除 这 些 之 外 ， 我 们 其 他 的 工作 都 与 
继承 极为 相似 。 下 面 是 乐器 例子 的 示意 图 : 




















Interface Instrument 


void play); 
String what); 
void adjust}; 














implements implements implements 


void play) void play) void play) 
String what) String whati) String what) 
void adjust) void adjust(} void adjust(} 











Brass 


void play) 
void adjust() 


其 体 实现 了 一 个 接口 以 后 ， 束 获得 了 一 个 普通 的 类 ， 可 用 标准 方式 对 其 
进行 扩展 。 


可 决定 将 一 个 接口 中 的 方法 声明 明确 定义 为 “public”*。 但 即便 不 明确 定 
义 ， 它 们 也 会 默认 为 public。 所 以 在 实现 一 个 接口 的 时 候 ， 来 自 接 口 的 
方法 必须 定义 成 public。 否 则 的 话 ， 它 们 会 默认 为 “友好 的 "， 而 且 会 限 
a Java ai Ea BS IC VET ABE 


在 Instrument 例 子 的 修改 版 本 中 ， 大 家 可 明确 地 看 出 这 一 点 。 注 意 接 口 
中 的 每 个 方法 都 严格 地 是 一 个 声明 ， 它 是 编译 器 唯一 允许 的 。 除 此 以 
外 ，Instrument5 中 没有 一 个 方法 被 声明 为 public， 但 它们 都 会 自动 获得 
public 属 性 。 如 下 所 示 : 


//: Music5.java 





// Interfaces 


import java.util.*; 


interface Instrument5 { 
// Compile-time constant: 

int i = 5; // static & final 

// Cannot have method definitions: 

void play(); // Automatically public 
String what(); 
void adjust(); 

} 

Class Wind5 implements Instrument5 { 
public void play() { 

System.out.printin("Wind5.play()"); 

} 
public String what() { return "Wind5"; } 
public void adjust() {} 

} 

class Percussion5 implements Instrument5 { 
public void play() { 

System.out.println("Percussion5.play()"); 

} 
public String what() { return "Percussion5"; } 
public void adjust() {} 


} 


class Stringed5 implements Instrument5 { 


public void play() { 
System.out.println("Stringed5.play()"); 
} 
public String what() { return "Stringed5"; } 
public void adjust() {} 
} 
class Brass5 extends Wind5 { 
public void play() { 
System.out.println("Brass5.play()"); 
} 
public void adjust() { 


System.out.println("Brass5.adjust()"); 


} 


Class Woodwind5 extends Wind5 { 
public void play() { 
System.out.println("Woodwind5.play()"); 


i 


public String what() { return "Woodwind5"; } 
} 
public class Music5 { 

// Doesn't care about type, so new types 


// added to the system still work right: 


static void tune(Instrument5 i) { 

PE aka 
i.play(); 

} 

static void tuneAll(Instrument5[] e) { 
for(int i = 0; i < e.length; i++) 

tune(e[i]); 

} 

public static void main(String[] args) { 
Instrument5[] orchestra = new Instrument5[5]; 
int i = 0; 
// Upcasting during addition to the array: 


orchestra[i++] = new Wind5(); 


orchestra[i++] = new Percussion5(); 


orchestra[it++] = new Stringed5(); 


orchestra[it+] new Brass5(); 
orchestra[it++] = new Woodwind5(); 
tuneAll(orchestra); 


} 
LV] i 





代码 剩余 的 部 分 按 相 同 的 方式 工作 。 我 们 可 以 上 自由 决定 上 渊 造型 到 一 个 
名 为 Instrument5 的 “普通 ”类 ， 一 个 名 为 nstrument5 的 “抽象 "类 ， 或 者 一 
个 名 为 Instrument5 的 “接口 ”。 所 有 行为 都 是 相同 的 。 事 实 上 ， 我 们 在 


tune() 方 法 中 可 以 发 现 没 有 任何 证 据 显 示 Instrument5 到 底 是 个 “ 普 
通 ” 类 、“ 抽 象 ” 类 还 是 一 个 “接口 "。 这 是 做 是 故意 的 : 每 种 方法 都 使 各 
序 员 能 对 对 象 的 创建 与 使 用 进行 不 同 的 控制 。 


7.5.1 Java 的 “多 重 继承 ” 


接口 只 是 比 抽象 类 “更 纯 ” 的 一 种 形式 。 它 的 用 途 并 不 止 那些 。 由 于 接口 
根本 没有 具体 的 实施 细 市 一 一 也 就 是 说 ， 没 有 与 存储 空间 与 “接口 ?关联 
在 一 起 一 一 所 以 没有 任何 办 法 可 以 防止 多 个 接口 合并 到 一 起 。 这 一 点 是 
至 关 重 要 的 ， 因 为 我 们 经 常 都 需要 表达 这 样 一 个 意思 : “从 属于 a， 也 
从 属于 b， 也 从 属于 c*。 在 C++ 中 ， 将 多 个 类 合并 到 一 起 的 行动 称 作 “ 多 
重 继承 ”， 而 且 操作 较为 不 便 ， 因 为 每 个 类 都 可 能 有 一 套 上 自己 的 实施 细 
节 。 在 Java 中 ， 我 们 可 采取 同样 的 行动 ， 但 只 有 其 中 一 个 类 拥有 具体 的 
实施 细 节 ， 所 以 在 全 并 多 个 接口 的 时 候 ，Cr+ 的 问题 不 会 在 ova 中 生 
o U PETR: 
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在 一 个 衍生 类 中 ， 我 们 并 不 一 定 要 拥有 一 个 抽象 或 具体 〈 没 有 抽象 方 
法 ) 的 基础 类 。 如 果 确 实 想 从 一 个 非 接口 继承 ， 那 么 只 能 从 一 个 继承 。 
和 独 余 的 所 有 基本 元 素 都 必须 是 “接口 "。 我 们 将 所 有 接口 名 置 于 
implements 关 键 字 的 后 面 ， 并 用 逗号 分 隔 它们 。 可 根据 需要 使 用 多 个 接 
口 ， 而 且 每 个 接口 都 会 成 为 一 个 独立 的 类 型 ， 可 对 其 进行 上 漳 造 型 。 下 
面 这 个 例子 展示 了 一 个 “具体 ”类 同 几 个 接口 合并 的 情况 ， 它 最 终生 成 了 


一 个 新 类 








K: 


//: Adventure.java 


// Multiple interfaces 


import java.util.*; 


interface CanFight { 
void fight(); 

} 

interface CanSwim { 
void swim(); 

} 

interface CanFly { 
void fly(); 

} 

class ActionCharacter { 
public void fight() {} 

} 

class Hero extends ActionCharacter 

implements CanFight, CanSwim, CanFly { 

public void swim() {} 
public void fly() {} 

} 

public class Adventure { 
static void t(CanFight x) { x.fight(); } 
static void u(CanSwim x) { x.Swim(); } 
static void v(CanFly x) { x.fly(); } 
static void w(ActionCharacter x) { x.fight(); } 


public static void main(String[] args) { 


Hero i = new Hero(); 

t(1); // Treat it as a CanFight 

u(i); // Treat it as a CanSwim 

v(i); // Treat it as a CanFly 

w(i); // Treat it as an ActionCharacter 


} 
Lif] l= 


从 中 可 以 看 到 ，Hero 将 具体 类 ActionCharacter 同 接口 CanFight， 
CanSwim 以 及 CanFly 合 并 起 来 。 按 这 种 形式 合并 一 个 具体 类 与 接口 的 时 
候 ， 有 具体 类 必须 首先 出 现 ， 然 后 才 是 接口 〈 人 否则 编译 器 会 报错 ) o 


请 注意 fight() 的 签名 在 CanFight 接 口 与 ActionCharacter 类 中 是 相同 的 ， 而 
且 没 有 在 Hero 中 为 fight() 提 供 一 个 具体 的 定义 。 接 口 的 规则 是 : 我 们 可 
以 从 和 它 继承 《〈 稍 后 就 会 看 到 ) ， 但 这 样 得 到 的 将 是 男 一 个 接口 。 如 果 想 
创建 新 类 型 的 一 个 对 象 ， 它 就 必须 是 已 提供 所 有 定义 的 一 个 类 。 尽 管 
Hero 没 有 为 fight() 明 确 地 提供 一 个 定义 ， 但 定义 是 随同 ActionCharacter 来 
的 ， 所 以 这 个 定义 会 自动 提供 ， 我 们 可 以 创建 Hero 的 对 象 。 


在 类 Adventure 中 ， 我 们 可 看 到 共有 四 个 方法 ， 它 们 将 不 同 的 接口 和 具 
体 类 作为 自己 的 自 变量 使 用 。 创 建 一 个 Hero 对 象 后 ， 它 可 以 传递 给 这 些 
方法 中 的 任何 一 个 。 这 意味 着 它们 会 依次 上 溯 造 型 到 每 一 个 接口 。 由 于 
接口 是 用 Java 设 计 的 ， 所 以 这 样 做 不 会 有 任何 问题 ， 而 且 程序 员 不 必 对 
此 加 以 任何 特别 的 关注 。 


注意 上 述 例子 已 向 我 们 揭示 了 接口 最 关键 的 作用 ， 也 是 使 用 接口 最 重要 
的 一 个 原因 : 能 上 漳 造 型 至 多 个 基础 类 。 使 用 接口 的 第 二 个 原因 与 使 用 
抽象 基础 类 的 原因 是 一 样 的 ; 防止 客户 程序 员 制 作 这 个 类 的 一 个 对 象 ， 

以 及 规定 它 仅 仅 是 一 个 接口 。 这 样 便 带 来 了 一 个 问题 ,到底 应 该 使 用 一 
个 接口 还 是 一 个 抽象 类 呢 ? 大 使 用 接口 ， 我 们 可 以 同时 获得 抽象 类 以 及 
接口 的 好 处 。 所 以 假如 想 创建 的 基础 类 没有 任何 方法 定义 或 者 成 员 变 

量 ， 那 么 无 论 如 何 都 愿意 使 用 接口 ， 而 不 要 选择 抽象 类 。 事 实 上 ， 如 果 
事先 知道 条 种 东西 会 成 为 基础 类 ， 那 么 第 一 个 选择 就 是 把 它 变 成 一 个 接 





























口 。 只 有 在 必须 使 用 方法 定义 或 者 成 员 变 量 的 时 候 ， 才 应 考虑 采用 抽象 


类 。 

7.5.2 通过 继承 扩展 接口 

利用 继承 技术 ， 可 方便 地 为 一 个 接口 添加 新 的 方法 声明 ， 也 可 以 将 几 个 
接口 合并 成 一 个 新 接口 。 在 这 两 种 情况 下 ， 最 终 得 到 的 都 是 一 个 新 接 
口 ， 如 下 例 所 示 : 


//: HorrorShow. java 





// Extending an interface with inheritance 
interface Monster { 
void menace(); 
} 
interface DangerousMonster extends Monster { 
void destroy(); 
} 
interface Lethal { 
void kill(); 
} 
class DragonZilla implements DangerousMonster { 
public void menace() {} 
public void destroy() {} 
} 
interface Vampire 


extends DangerousMonster, Lethal { 


void drinkBlood(); 
} 
class HorrorShow { 
static void u(Monster b) { b.menace(); } 
static void v(DangerousMonster d) { 
d.menace(); 
d.destroy(); 
} 
public static void main(String[] args) { 
DragonZilla if2 = new DragonZilla(); 
u(if2); 
v(if2); 
} 
ALLS 


DangerousMonster 是 对 Monster 的 一 个 简单 的 扩展 ， 最 终生 成 了 一 个 新 接 
口 。 这 是 在 DragonZilla 里 实现 的 。 


Vampire 的 语法 仅 在 继承 接口 时 才 可 使 用 。 通 第 ， 我 们 只 能 对 单独 一 个 
类 应 用 extends (扩展 ) 关键 字 。 但 由 于 接口 可 能 由 多 个 其 他 接口 构成 ， 
所 以 在 构建 一 个 新 接口 时 ，extends 可 能 引用 多 个 基础 接口 。 正 如 大 家 看 
到 的 那样 ， 接 口 的 名 字 只 是 简单 地 使 用 逗号 分 隔 。 


7.5.3 常数 分 组 
由 于 置 入 一 个 接口 的 所 有 字段 都 自动 具有 static 和 final 属 性 ， 所 以 接口 是 


对 第 数值 进行 分 组 的 一 个 好 工具 ， 它 具有 与 C 或 C++ 的 enum 非 党 相似 的 
效果 。 如 下 例 所 示 : 














//: Months.java 


// Using interfaces to create groups of constants 

package c07; 

public interface Months { 

int 

JANUARY = 1, FEBRUARY = 2, MARCH = 3, 
APRIL = 4, MAY = 5, JUNE = 6, JULY = 7, 
AUGUST = 8, SEPTEMBER = 9, OCTOBER = 10, 
NOVEMBER = 11, DECEMBER = 12; 


E 


注意 根据 Java 命 名 规则 ， 拥 有 固定 标识 符 的 static final 基 本 数据 类 型 OR 
i 
上 单词 ) 。 


接口 中 的 字段 会 自动 具备 public 属 性 ， 所 以 没 必要 专门 指定 。 


现在 ， 通 过 导入 c07.* 或 c07.Months， 我 们 可 以 从 包 的 外 部 使 用 常数 一 
就 象 对 其 他 任何 包 进 行 的 操作 那样 。 此 外 ， 也 可 以 用 类 似 
Months.JANUARY 的 表达 式 对 值 进行 引用 。 当 然 ， 我 们 获得 的 只 是 一 个 
int， 所 以 不 象 C++ 的 enum 那 样 拥有 额外 的 类 型 安全 性 。 但 与 将 数字 强行 
编码 〈 硬 编码 ) 到 自己 的 程序 中 相 比 ， 这 种 (常用 的 ) 技术 无 疑 已 经 是 
一 个 巨大 的 进步 。 我 们 通常 把 “人 硬 编 码 ” 数 字 的 行为 称 为 “魔术 数字 ”， 它 
产生 的 代码 是 非常 难以 维护 的 。 


A eres 的 类 型 安全 性 ， 可 构建 象 下 面 这 样 的 一 个 类 (注释 








//: Month2.java 


// A more robust enumeration system 

package c07; 

public final class Month2 { 
private String name; 
private Month2(String nm) { name = nm; } 
public String toString() { return name; } 
public final static Month2 


JAN 


new Month2("January"), 

FEB = new Month2("February"), 

MAR = new Month2("March"), 

APR = new Month2("April"), 

MAY = new Month2("May"), 

JUN = new Month2("June"), 

JUL = new Month2("July"), 

AUG = new Month2("August"), 

SEP = new Month2("September"), 

OCT = new Month2("October"), 

NOV = new Month2("November"), 

DEC = new Month2("December"); 
public final static Month2[] month = { 

JAN, JAN, FEB, MAR, APR, MAY, JUN, 


JUL, AUG, SEP, OCT, NOV, DEC 


3 


public static void main(String[] args) { 
Month2 m = Month2. JAN; 
System.out.println(m); 
m = Month2.month[12]; 
System.out.println(m); 
System.out.println(m == Month2.DEC); 
System.out.println(m.equals(Month2.DEC)); 

} 

} ///:~ 


D: 是 Rich Hoffarth 的 一 封 E-mail 触 发 了 我 这 样 编写 程序 的 灵感 。 


这 个 类 叫 作 Month2， 因 为 标准 Java 库 里 已 经 有 一 个 Month。 它 是 一 个 
final 类 ， 并 含有 一 个 private 构 建 器 ， 所 以 没有 人 能 从 它 继承 ， 或 制作 筷 
的 一 个 实例 。 唯 一 的 实例 就 是 那些 final static 对 象 ， 它 们 是 在 类 本 身 内 部 
创建 的 ， 包 括 : JAN，FEB，MAR 等 等 。 这 些 对 象 也 在 month 数 组 中 使 
用 ， 后 者 让 我 们 能 够 按 数 字 挑 选 月 份 ， 而 不 是 按 名 字 〈 注 意 数组 中 提供 
了 一 个 多 余 的 JAN， 使 偏 移 量 增加 了 1， 也 使 December 确 实 成 为 12 

H) 。 在 main0 中 ， 我 们 可 注意 到 类 型 的 安全 性 : m 是 一 个 Month2 对 
象 ， 所 以 只 能 将 其 分 配给 Month2。 在 前 面 的 Months.java 例 子 中 ， 只 提供 
了 int 值 ， 所 以 本 来 想 用 来 代表 一 个 月 份 的 int 变 量 可 能 实际 获得 一 个 整数 
值 ， 那 样 做 可 能 不 十 分 安全 。 


这 儿 介 绍 的 方法 也 人 允许 我 们 交换 使 用 == 或 者 equals0， 就 象 main0 尾 部 展 
示 的 那样 。 


7.5.4 初始 化 接口 中 的 字段 


接口 中 定义 的 字段 会 自动 具有 static 和 final 属 性 。 它 们 不 能 是 “空白 
final*， 但 可 初始 化 成 非常 数 表 达 式 。 例 如 : 


//: RandVals.java 














// Initializing interface fields with 

// non-constant initializers 

import java.util.*; 

public interface RandVals { 
int rint = (int)(Math.random() * 10); 
long rlong = (long)(Math.random() * 10); 
float rfloat = (float)(Math.random() * 10); 
double rdouble = Math.random() * 10; 


A 








由 于 字段 是 static 的 ， 所 以 它们 会 在 首次 装载 关 之 后 、 以 及 首次 访问 任何 
字段 之 前 获得 初始 化 。 下 面 是 一 个 简单 的 测试 : 


//: TestRandVals.java 








public class TestRandVals { 
public static void main(String[] args) { 
System.out.println(RandVals.rint); 
System.out.println(RandVals.rlong); 
System.out.println(RandVals.rfloat); 


System.out.printin(RandVals.rdouble); 


} ///:~ 





A 字段 并 不 是 接口 的 一 部 分 ， 而 是 保存 于 那个 接口 的 static 存 储 区 域 


7.6 内 部 类 


在 Java ”1.1 中， 可 将 一 个 类 定义 置 入 男 一 个 类 定义 中 。 这 就 叫 作 “ 内 部 
类 ”。 内 部 类 对 我 们 非常 有 用 ， 因 为 利用 E HI AJ AB Eee EEL AS EN 

VT 分 组 ， 并 可 控制 一 个 大 在 另 一 个 类 里 的 “可 见 性 ?。 然 而 ， 我 们 必 
须 认识 到 内 部 类 与 以 前 讲述 的 “合成 "方法 存在 着 根本 的 区 别 ， 

通常 ， 对 内 部 类 的 需要 并 不 是 特 Aa 至 少 不 会 立即 感觉 到 目 己 需 
要 使 用 内 部 类 。 在 本 章 的 末尾 ， 介 绍 完 内 部 类 的 所 有 语法 之 后 ， 大 家 会 
发 现 一 个 特别 的 例子 。 通 过 它 应 该 可 以 清晰 地 认识 到 内 部 类 的 好 处 。 


创建 内 部 类 的 过 程 是 平淡 无 奇 的 :将 类 定义 置 入 一 个 用 于 封装 它 的 类 内 
部 〈 若 执行 这 个 程序 遇 到 麻烦 ， 请 参见 第 3 章 的 3.1.2 小 节 “ 赋 值 ?>) : 


//: Parceli.java 














// Creating inner classes 
package c07.parcelti; 
public class Parceli1 { 
class Contents { 
private int i = 11; 
public int value() { return i; } 
小 
class Destination { 
private String label; 
Destination(String whereTo) { 


label = whereTo; 


String readLabel() { return label; } 

} 

// Using inner classes looks just like 
// using any other class, within Parceli: 
public void ship(String dest) { 

Contents c = new Contents(); 
Destination d = new Destination(dest); 
} 
public static void main(String[] args) { 
Parcel1 p = new Parceli(); 
p.ship("Tanzania"); 
} 
} ///:~ 


在 在 shipO 内 部 使 用 ， 内 部 类 的 使 用 看 起 来 和 其 他 任何 类 都 没什么 分 

别 。 在 这 里 ， 唯 一 明显 的 区 别 融 是 它 的 名 字 骨 套 在 Parcel1 里 面 。 但 大 家 
不 久 就 会 知道 ， 这 其 实 并 非 唯一 的 区 别 。 

更 典型 的 一 种 情况 是 ， 一 个 外 部 类 拥有 一 个 特殊 的 方法 ， 它 会 返回 指 问 
一 个 内 部 类 的 句柄 。 束 象 下 面 这 样 : 


//: Parcel2.java 





// Returning a handle to an inner class 
package c07.parcel2; 


public class Parcel2 { 


class Contents { 
private int i = 11; 
public int value() { return i; } 
} 
class Destination { 
private String label; 
Destination(String whereTo) { 
label = whereTo; 


} 


String readLabel() { return label; } 
} 
public Destination to(String s) { 
return new Destination(s); 
} 
public Contents cont() { 
return new Contents(); 
} 
public void ship(String dest) { 
Contents c = cont(); 
Destination d = to(dest); 
} 
public static void main(String[] args) 


Parcel2 p = new Parcel2(); 


p.ship("Tanzania"); 
Parcel2 q = new Parcel2(); 
// Defining handles to inner classes: 
Parcel2.Contents c = q.cont(); 
Parcel2.Destination d = q.to("Borneo"); 
} 
A 


硅 想 在 除外 部 类 非 static 方 法 内 部 之 外 的 任何 地 方 生成 内 部 类 的 一 个 对 
ee re ee a nee 
示 的 那样 。 


7.6.1 内 部 类 和 上 渊 造型 


迄今 为 目 ， 内 部 类 看 起 来 仍然 没什么 特别 的 地 方 。 毕 竟 ， 用 它 实现 隐藏 
显得 有 些 大 题 小 做 。Java 已 经 有 一 个 非常 优秀 的 隐藏 机 制 一 一 只 允许 类 
成 为 "友好 的 ”《〈 只 在 一 个 包 内 可 见 ) ， 而 不 是 把 它 创 建成 一 个 内 部 类 。 


然而 ， 当 我 们 准备 上 调 造 型 到 一 个 基础 类 《特别 是 到 一 个 接口 ) 的 时 

候 ， 内 部 类 就 开始 发 挥 其 关键 作用 《〈 从 用 于 实现 的 对 象 生成 一 个 接口 名 
柄 具有 与 上 调 造 型 至 一 个 基础 类 相同 的 效果 ) 。 这 是 由 于 内 部 类 随后 可 
完全 进入 不 可 见 或 不 可 用 状态 一 一 对 任何 人 都 将 如 此 。 所 以 我 们 可 以 非 
常 方 便 地 隐藏 实施 细节 。 我 们 得 到 的 全 部 回报 就 是 一 个 基础 类 或 者 接口 
的 句柄 ， 而 且 其 至 有 可 能 不 知道 准确 的 类 型 。 就 象 下 面 这 样 : 


//: Parcel3.java 























// Returning a handle to an inner class 
package c07.parcel3; 
abstract class Contents { 


abstract public int value(); 


j 


interface Destination { 
String readLabel(); 
} 
public class Parcel3 { 
private class PContents extends Contents { 
private int i = 11; 
public int value() { return i; } 
} 
protected class PDestination 
implements Destination { 
private String label; 
private PDestination(String whereTo) { 
label = whereTo; 
} 
public String readLabel() { return label; } 
} 
public Destination dest(String s) { 
return new PDestination(s); 
} 
public Contents cont() { 


return new PContents(); 


} 
class Test { 
public static void main(String[] args) { 
Parcel3 p = new Parcel3(); 
Contents c = p.cont(); 
Destination d = p.dest("Tanzania"); 
// Illegal -- can't access private class: 
//! Parcel3.PContents c = p.new PContents(); 
} 
} ///:~ 








现在 ，Contents 和 Destination 代 表 可 由 客户 程序 员 使 用 的 接口 〈 记 住 接口 
会 将 上 自己 的 所 有 成 员 都 变 成 public 属 性 ) 。 为 方便 起 见 ， 它 们 置 于 单独 
一 个 文件 里 ， 但 原始 的 Contents 和 Destination 在 它们 自己 的 文件 中 是 相互 
public 的 。 


在 Parcel3 中 ， 一 些 新 东西 已 经 加 入 : 内 部 类 PContents 被 设 为 private， 所 
以 除了 Parcel3 之 外 ， 其 他 任何 东西 都 不 能 访问 它 。PDestination 被 设 为 

protected， 所 以 除了 Parcel3，Parcel3 包 内 的 类 【因为 protected 也 为 包 赋 
予 了 访问 权 ; 也 就 是 说 ，protected 也 是 “友好 的 ”) ， 以 及 Parcel3 的 继承 
者 之 外 ， 其 他 任何 东西 都 不 能 访问 PDestination。 这 意味 着 客户 程序 员 对 
这 些 成 员 的 认识 与 访问 将 会 受到 限制 。 事 实 上 ， 我 们 甚至 不 能 下 淹 造 型 
到 一 个 private 内 部 类 〈 或 者 一 个 protected 内 部 类 ， 除 非 自 己 本 身 便 是 一 
个 继承 者 ) ， 因 为 我 们 不 能 访问 名 字 ， 就 象 在 classTest 里 看 到 的 那样 。 

所 以 ， 利 用 private 内 部 类 ， 类 设计 人 员 可 完全 禁止 其 他 人 依赖 类 型 编 

人 码 ， 并 可 将 具体 的 实施 细节 完全 隐藏 起 来 。 除 此 以 外 ， 从 客户 程序 员 的 
角度 来 看 ， 一 个 接口 的 范围 没有 意义 的 ， 因 为 他 们 不 能 访问 不 属于 公共 
这 样 一 来 ，Java 编 译 占 也 有 机 会 生成 效率 更 高 


普通 〈 非 内 部 ) 类 不 可 设 为 private 或 protected 

















只 人 允许 public 或 者 “ 友 





好 的 ”。 


注意 Contents 不 必 成 为 一 个 抽象 类 。 在 这 儿 也 可 以 使 用 一 个 普通 类 ， 但 
这 种 设计 最 典型 的 起 点 依然 是 一 个 “接口 ”。 

7.6.2 方法 和 作用 域 中 的 内 部 类 

至 此 ， 我 们 已 基本 理解 了 内 部 类 的 典型 用 途 。 对 那些 涉及 内 部 类 的 代 

码 ， 通 常 表达 的 都 是 “单纯 ”的 内 部 类 ， 非 党 简单 ， 且 极 易 理解 。 然 而 ， 

内 部 类 的 设计 非常 全 面 ， 不 可 避免 地 会 遇 到 它们 的 其 他 大 量 用 法 一 一 假 
知 我 们 在 一 个 方法 甚至 一 个 任意 的 作用 域内 创建 内 部 类 。 有 两 方面 的 原 
因 促使 我 们 这 样 做 : 


(1) 正如 前 面 展 示 的 那样 ， 我 们 准备 实现 东 种 形式 的 接口 ， 使 目 己 能 创 
建 和 返回 一 个 句柄 。 


(2) ”要 解决 一 个 复杂 的 问题 ， 并 希望 创建 一 个 类 ， 用 来 辅助 自己 的 程序 
方案 。 同 时 不 愿意 把 它 公 开 。 


在 下 面 这 个 例子 里 ， 将 修改 前 面 的 代码 ， 以 便 使 用 : 

(1) 在 一 个 方法 内 定义 的 类 

(2) 在 方法 的 一 个 作用 域内 定义 的 类 

(3) 一 个 匿名 类 ， 用 于 实现 一 个 接口 

(4) 一 个 匿名 类， 用 于 扩展 拥有 非 默认 构建 锅 的 一 个 类 

(5) 一 个 匿名 类， 用 于 执行 字段 初始 化 
人 





所 有 这 些 都 在 innerscopes 包 内 发 生 。 首 先 ， 来 目前 述 代 码 的 通用 接口 会 
在 它们 自己 的 文件 里 获得 定义 ， 使 它们 能 在 所 有 的 例子 里 使 用 : 


//: Destination.java 


package c07.innerscopes; 
interface Destination { 
String readLabel(); 


} ///:~ 


oma 已 认为 Contents 可 能 是 一 个 抽象 类 ， 


然 的 形式 ， 束 象 一 个 接口 那样 : 


//: Contents, java 


package c07.innerscopes; 
interface Contents { 
int value(); 


} ///:~ 











尽管 是 含有 具体 实施 细 证 的 一 个 普通 
类 的 一 个 通用 “接口 ”使 用 : 


//: Wrapping.java 


package c07.innerscopes; 
public class Wrapping { 
private int i; 


public Wrapping(int x) { i 


public int value() { return i; 


} ///:~ 





所 以 可 采取 下 面 这 种 更 目 


， 但 Wrapping 也 作为 它 所 有 衍生 


X; } 
} 


在 上 面 的 代码 中 ， 我 们 注意 到 Wrapping 有 一 个 要 求 使 用 自 变 量 的 构建 
fit, TQ ETS OLS Fo ENA 了 。 

第 一 个 例子 展示 了 如 何在 一 个 方法 的 作用 域 〈 而 不 是 另 一 个 类 的 作用 
域 ) 中 创建 一 个 完整 的 类 : 


//: Parcel4.java 


// Nesting a class within a method 
package c07.innerscopes; 
public class Parcel4 { 
public Destination dest(String s) { 
class PDestination 
implements Destination { 
private String label; 
private PDestination(String whereTo) { 
label = whereTo; 
} 
public String readLabel() { return label; } 
} 
return new PDestination(s); 
} 
public static void main(String[] args) { 
Parcel4 p = new Parcel4(); 


Destination d = p.dest("Tanzania"); 


} ///:~ 


PDestination 类 属于 dest() 的 一 部 分 ， 而 不 是 Parcel4 的 一 部 分 (同时 注意 
可 为 相同 目录 内 每 个 类 内 部 的 一 个 内 部 类 使 用 类 标识 符 PDestination， 这 
样 做 不 会 发 生命 名 的 冲突 ) 。 因 此 ，PDestination 不 可 从 dest() 的 外 部 访 
问 。 请 注意 在 返回 语句 中 发 生 的 上 漳 造 型 除了 指 问 基础 类 
Destination 的 一 个 句柄 之 外 ， 没 有 任何 东西 超出 destO 的 边界 之 外 。 当 
然 ， 不 能 由 于 类 PDestination 的 名 字 置 于 destO 内 部 ， 束 认为 在 dest0 返 回 
之 后 PDestination 不 是 一 个 有 效 的 对 象 。 


下 面 这 个 例子 展示 了 如 何在 任意 作用 域内 藤 套 一 个 内 部 类 : 


//: Parcel5.java 








// Nesting a class within a scope 
package c0O7.innerscopes; 
public class Parcel5d { 
private void internalTracking(boolean b) { 
if(b) { 
class TrackingSlip { 
private String id; 
TrackingSlip(String s) { 
id = s; 
} 
String getSlip() { return id; } 
} 
TrackingSlip ts = new TrackingSlip("slip"); 


String s = ts.getSlip(); 


} 
// Can't use it here! Out of scope: 
//! TrackingSlip ts = new TrackingSlip("x"); 
} 
public void track() { internalTracking(true); } 
public static void main(String[] args) { 
Parcel5 p = new Parcel5(); 
p.track(); 
} 
A Ar 




















ee 句 的 作用 域内 。 这 并 不 意味 看 类 是 有 条 件 
它 会 随同 其 他 所 有 东西 得 到 编译 。 然 而 ， 在 定义 它 的 那个 作 
ER REED SEES 和 一 个 普通 类 并 没 
有 什么 区 别 。 


下 面 这 个 例子 看 起 来 有 些 奇 怪 : 


//: Parcel6.java 














// A method that returns an anonymous inner class 
package c07.innerscopes; 
public class Parcel6é { 
public Contents cont() { 
return new Contents() { 


private int i = 11; 


public int value() { return i; } 
}; // Semicolon required in this case 
} 
public static void main(String[] args) { 
Parcel6 p = new Parcel6(); 
Contents c = p.cont(); 


} 
SA 


cont() 方 法 同时 合并 了 返回 值 的 创建 代码 ， 以 及 用 于 表示 那个 返回 值 的 
类 。 除 此 以 外 ， 这 个 类 是 匿名 的 一 一 它 没 有 名 字 。 而 且 看 起 来 似乎 更 让 
人 摸 不 着 头脑 的 是 ， 我 们 准备 创建 一 个 Contents 对 象 : 





return new Contents() 


但 在 这 之 后 ， 在 遇 到 分 号 之 前 ， 我 们 又 说 :“ 等 一 等 ， 让 我 先 在 一 个 类 
定义 里 再 要 一 下 花招 ”: 


return new Contents() { 

private int i = 11; 

public int value() { return i; } 

} 

APTER OA ee: “AE MA ContentshiT E H RAE 4 


一 个 对 象 ”。 由 new 表 达 式 返回 的 句柄 会 目 动 上 渊 造型 成 一 个 Contents 人 句 
柄 。 匿 名 内 部 类 的 语法 其 实 要 表达 的 是 : 








class MyContents extends Contents { 


private int i = 11; 


public int value() { return i; } 
} 
return new MyContents(); 


FEE A AEBS, Contents xe H — ARNA EREA. RTE EPS 
FRAN J SERIE tis EA BOE Sd Sa EY A ST : 


//: Parcel7.java 





// An anonymous inner class that calls the 
// base-class constructor 
package c07.innerscopes; 
public class Parcel7 { 
public Wrapping wrap(int x) { 
// Base constructor call: 
return new Wrapping(x) { 
public int value() { 
return super.value() * 47; 
} 
}; // Semicolon required 
} 
public static void main(String[] args) { 
Parcel7 p = new Parcel7(); 


Wrapping w = p.wrap(10); 


} ///:~ 


也 束 是 说 ， 我 们 将 适当 的 自 变 量 简单 地 传递 给 基础 类 构建 器 ， 在 这 儿 表 
现 为 在 “new Wrapping(x)” 中 传递 x。 匿 名 类 不 能 拥有 一 个 构建 器 ， 这 和 
在 调用 superO 时 的 常规 做 法 不 同 。 

在 前 述 的 两 个 例子 中 ， 分 号 并 不 标志 着 类 主体 的 结束 〈 和 C++ 不 同 ) 。 
相反 ， 它 标志 着 用 于 包含 匿名 类 的 那个 表达 式 的 结束 。 因 此 ， 它 完全 等 
价 于 在 其 他 任何 地 方 使 用 分 号 。 

若 想 对 匿名 内 部 类 的 一 个 对 象 进行 某 种 形式 的 初始 化 ， 此 时 会 出 现 什么 
情况 呢 ? 由 于 它 是 匿名 的 ， 没 有 名 字 赋 给 构建 费 ， 所 以 我 们 不 能 拥有 一 
个 构建 器 。 然 而 ， 我 们 可 在 定义 自己 的 字段 时 进行 初始 化 : 


//: Parcel8.java 








// An anonymous inner class that performs 
// initialization. A briefer version 
// of Parcel5.java. 
package c07.innerscopes; 
public class Parcels { 
// Argument must be final to use inside 
// anonymous inner class: 
public Destination dest(final String dest) { 
return new Destination() { 
private String label = dest; 
public String readLabel() { return label; } 


}; 


public static void main(String[] args) { 
Parcel8 p = new Parcel8(); 
Destination d = p.dest("Tanzania"); 


i 


} ///:~ 





各 试图 定义 一 个 匿名 内 部 类 ， 并 想 使 用 在 匿名 内 部 类 外 部 定义 的 一 个 对 
象 ， 则 编译 占 要 求 外 部 对 象 为 final 属 性 。 这 正 是 我 们 将 dest() 的 目 变 量 设 
为 final 的 原因 。 如 果 环 记 这 样 做 ， 束 会 得 到 一 条 编译 期 出 错 提示 。 

只 要 上 自己 只 是 想 分 配 一 个 字段 ， 上 述 方法 就 肯定 可 行 。 但 假如 需要 采取 
一 些 类 似 于 构建 锅 的 行动 ， 勾 应 怎样 操作 呢 ? 通过 Java ”1.1 的 实例 初始 
化 ， 我 们 可 以 有 效 地 为 一 个 匿名 内 部 类 创建 一 个 构建 器 : 


//: Parcel9.java 








// Using "instance initialization" to perform 
// construction on an anonymous inner class 
package c0O7.innerscopes; 
public class Parcel9 { 
public Destination 
dest(final String dest, final float price) { 
return new Destination() { 
private int cost; 
// Instance initialization for each object: 


{ 


cost = Math.round(price); 


if(cost > 100) 
System.out.println("Over budget!"); 
} 
private String label = dest; 
public String readLabel() { return label; } 
}; 
} 
public static void main(String[] args) { 
Parcel9 p = new Parcel9(); 
Destination d = p.dest("Tanzania", 101.395F); 
} 
} ///:~ 


在 实例 初始 化 模 卖 中 ， 我 们 可 看 到 代码 不 能 作为 类 初始 化 模块 〈 即 站 语 
AJ) 的 一 部 分 执行 。 所 以 实际 上 ， 一 个 实例 初始 化 模块 就 是 一 个 匿名 内 
部 类 的 构建 器 。 当 然 ， 筷 的 功能 是 有 限 的 ;我 们 不 能 对 实例 初始 化 模块 
进行 过 载 处 理 ， 所 以 只 能 拥有 这 些 构建 器 的 其 中 一 个 。 


7.6.3 链接 到 外 部 类 


运 今 为 止 ， 我 们 见 到 的 内 部 类 好 象 仅仅 是 一 种 名 字 隐 藏 以 及 代码 组 织 方 
和 案 。 尽 管 这 些 功 能 非常 有 用 ， 但 似乎 并 不 特别 引 人 注 目 。 然 而 ， 我 们 还 
忽略 了 男 一 个 重要 的 事实 。 创 建 自己 的 内 部 类 时 ， 那 个 类 的 对 象 同时 拥 
有 指 同 封 朔 对象 〈 这 些 对 象 封 装 或 生成 了 内 部 类 ) 的 一 个 链接 。 所 以 它 
们 能 访问 那个 封装 对 象 的 成 员 一 一 堵 需 取得 任何 资格 。 除 此 以 外 ， 内 部 
类 拥有 对 封装 类 所 有 元 素 的 访问 权限 (注释) 。 下 面 这 个 例子 阐 示 了 


这 个 问题 : 














//: Sequence.java 


// Holds a sequence of Objects 
interface Selector { 
boolean end(); 
Object current(); 
void next(); 
} 
public class Sequence { 
private Object[] o; 
private int next = 0; 
public Sequence(int size) { 
o = new Object[size]; 
} 
public void add(Object x) { 


if(next < o.length) { 


o[next ] xX; 


next++; 


} 

private class SSelector implements Selector { 
int i = 0; 
public boolean end() { 


return i == o.length; 


public Object current() { 
return o[1]; 

J 

public void next() { 


if(i < o.length) i++; 


} 


public Selector getSelector() { 
return new SSelector(); 
} 
public static void main(String[] args) { 
Sequence s = new Sequence(10); 
for(int i = 0; i < 10; i++) 
s.add(Integer.toString(i)); 
Selector sl = s.getSelector(); 
while(!sl.end()) { 
System.out.printin((String)sl.current()); 


sl.next(); 


} 
} MA i~ 


D: HCH REREAD i 


Re 
又 限 。 


其 中 ，S$equence 只 是 一 个 大 小 固定 的 对 象 数组 ， 有 一 个 类 将 其 封装 在 内 
部 。 我 们 调用 add0， 以 便 将 一 个 新 对 象 添 加 到 Sequence 末 尾 〈 如 果 还 有 
地 方 的 话 ) 。 为 了 取得 Sequence 中 的 每 一 个 对 象 ， 要 使 用 一 个 名 为 
Selector 的 接口 ， 它 使 我 们 能 够 知道 目 己 是 否 位 于 最 末尾 (end0) , fë 
观看 当前 对 象 Ccurrent() Object) ， 以 及 能 够 移 至 Sequence 内 的 下 一 个 
对 象 (next() Object) 。 由 于 Selector 是 一 个 接口 ， 所 以 其 他 许多 类 都 能 
用 它们 自己 的 方式 实现 接口 ， 而 且 许 多 方法 都 能 将 接口 作为 一 个 自 变 量 
使 用 ， 从 而 创建 一 般 的 代码 。 


在 这 里 ，SSelector 是 一 个 私有 类 ， 它 提供 了 Selector 功 能 。 在 main0 中 ， 
大 家 可 看 到 Sequence 的 创建 过 程 ， 在 它 后 面 是 一 系列 字 串 对 象 的 添加 。 
随后 ， 通 过 对 getSelector() 的 一 个 调用 生成 一 个 Selector。 并 用 它 在 
Sequence 中 移动 ， 同 时 选择 每 一 个 项 目 。 


从 表面 看 ，SSelector 似 乎 只 是 另 一 个 内 部 类 。 但 不 要 被 表面 现象 迷惑 。 
请 注意 观察 end0)，currentO 以 及 next0， 它 们 每 个 方法 都 引用 了 o。o 是 个 
不 属于 SSelector 一 部 分 的 句柄 ， 而 是 位 于 封装 类 里 的 一 个 private 字 段 。 
然而 ， 内 部 类 可 以 从 封装 类 访问 方法 与 字段 ， 就 象 已 经 拥有 了 它们 一 
wa aaa 




















因此 ， 我 们 现在 知道 一 个 内 部 类 可 以 访问 封装 类 的 成 员 。 这 是 如 何 实现 
的 呢 ? 内 部 类 必须 拥有 对 封闭 类 的 特定 对 象 的 一 个 引用 ， 而 封装 类 的 作 
用 就 是 创建 这 个 内部 类 。 随 后 ， 当 我 们 引用 封装 类 的 一 个 成 员 时 ， 就 利 
用 那个 “隐藏 ;的 引用 来 选择 那个 成 员 。 竺 运 的 是 ， 编 译 器 会 帮助 我 们 
照管 所 有 这 些 细 市 。 但 我 们 现在 也 可 以 理解 内 部 类 的 一 个 对 象 只 能 与 封 
装 类 的 一 个 对 象 联合 创建 。 在 这 个 创建 过 程 中 ， 要 求 对 封装 类 对 象 的 句 
柄 进行 初始 化 。 知 不 能 访问 那个 句柄 ， 编 译 需 就 会 报错 。 进 行 所 有 这 些 
操作 的 时 候 ， 大 多 数 时 候 都 不 要 求 程序 员 的 任何 介入 。 


7.6.4 static 内 部 类 
为 正确 理解 static 在 应 用 于 内 部 类 时 的 含义 ， 必 须 记 住 内 部 类 的 对 象 默 认 


持 有 创建 它 的 那个 封 厄 类 的 一 个 对 象 的 句柄 。 然 而 ， 假 如 我 们 说 一 个 内 
部 类 是 static 的 ， 这 种 说 法 却 是 不 成 并 的 。static 内 部 类 意味 着 : 





(1) 为 创建 一 个 static 内 部 类 的 对 象 ， 我 们 不 需要 一 个 外 部 类 对 象 。 
(2) 不 能 从 static 内 部 类 的 一 个 对 象 中 访问 一 个 外 部 类 对 象 。 


但 在 存在 一 些 限 制 : 由 于 static 成 员 只 能 位 于 一 个 类 的 外 部 级 别 ， 所 以 内 
部 类 不 可 拥有 static 数 据 或 static 内 部 类 。 

倘若 为 了 创建 内 部 类 的 对 象 而 不 需要 创建 外 部 类 的 一 个 对 象 ， 那 么 可 将 
所 有 东西 都 设 为 static。 为 了 能 正常 工作 ， 同 时 也 必须 将 内 部 类 设 为 
static. UR Ata: 


//: Parceli0.java 


// Static inner classes 
package c07.parcel10; 
abstract class Contents { 
abstract public int value(); 
} 
interface Destination { 
String readLabel(); 
} 
public class Parcel10 { 
private static class PContents 
extends Contents { 
private int i = 11; 
public int value() { return i; } 


} 


protected static class PDestination 


implements Destination { 
private String label; 
private PDestination(String whereTo) { 
label = whereTo; 
} 
public String readLabel() { return label; } 
小 
public static Destination dest(String s) { 
return new PDestination(s); 
} 
public static Contents cont() { 
return new PContents(); 
小 
public static void main(String[] args) { 
Contents c = cont(); 
Destination d = dest("Tanzania"); 
} 
} ///:~ 


在 main() 中 ， 我 们 不 需要 Parcel10 的 对 象 ;， 相反， 我 们 用 常规 的 语法 来 选 
择 一 个 static 成 员 ， 以 便 调 用 将 句柄 返回 Contents 和 Destination 的 方法 。 


通常 ， 我 们 不 在 一 个 接口 里 设置 任何 代码 ， 但 static 内 部 类 可 以 成 为 接口 
的 一 部 分 。 由 于 类 是 “静态 ”的 ， 所 以 它 不 会 违反 接口 的 规则 一 一 static 内 
部 类 只 位 于 接口 的 命名 空间 内 部 : 





//: TInterface.java 


// Static inner classes inside interfaces 
interface IInterface { 
static class Inner { 
int i, j, k; 
public Inner() {} 
void f() {} 


} 
RTs 
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不 愿 如 此 ， 可 考虑 用 一 个 static 内 部 类 容纳 目 己 的 测试 代码 。 如 下 所 示 : 


//: TestBed.java 





// Putting test code in a static inner class 
class TestBed { 
TestBed() {} 
void f() { System.out.println("f()"); } 
public static class Tester { 
public static void main(String[] args) { 
TestBed t = new TestBed(); 


t.f(); 


} 
于 LA 


这 样 便 生成 一 个 独立 的 、 名 为 TestBed$Tester 的 类 (为 运行 程序 ， 请 使 
用 “java TestBed$Tester’a74) 。 可 将 这 个 类 用 于 测试 ， 但 不 需 在 自己 的 
最 终 发 行 版 本 中 包含 它 。 


7.6.5 引用 外 部 类 对 象 


香 想 生成 外 部 类 对 象 的 句柄 ， 就 要 用 一 个 点 号 以 及 一 个 this 来 命名 外 部 

类 。 举 个 例子 来 说 ， 在 Sequence.SSelector 类 中 ， 它 的 所 有 方法 都 能 产生 
外 部 类 Sequence 的 存储 句柄 ， 方 法 是 采用 Sequence.this 的 形式 。 结 果 获 

得 的 句柄 会 自动 具备 正确 的 类 型 〈 这 会 在 编译 期 间 检 查 并 核实 ， 所 以 不 
会 出 现 运 行 期 的 开销 ) 。 

有 些 时 候 ， 我 们 想 告 诉 其 他 某 些 对 象 创建 它 某 个 内 部 类 的 一 个 对 象 。 为 
达到 这 个 目的 ， 必 须 在 new 表 达 式 中 提供 指 癌 其 他 外 部 类 对 象 的 一 个 句 
柄 ， 就 象 下 面 这 样 : 


//: Parcel11.java 











// Creating inner classes 
package c07.parcelii1; 
public class Parcelii { 
class Contents { 
private int i = 11; 
public int value() { return i; } 
J 
class Destination { 


private String label; 


Destination(String whereTo) { 
label = whereTo; 
} 
String readLabel() { return label; } 
} 
public static void main(String[] args) { 
Parcelii p = new Parceli1(); 
// Must use instance of outer class 
// to create an instances of the inner class: 
Parceli1.Contents c = p.new Contents(); 
Parceli1.Destination d = 
p.new Destination("Tanzania"); 
小 


} ///:~ 


为 直接 创建 内 部 类 的 一 个 对 象 ， 不 能 象 大 家 或 许 猜想 的 那样 一 一 采用 相 
同 的 形式 ， 并 引用 外 部 类 名 Parcel11。 此 时 ， 必 须 利用 外 部 类 的 一 个 对 
象 生 成 内 部 类 的 一 个 对 象 : 





Parcel11.Contents c = p.new Contents(); 

因此 ， 除 非 已 拥有 外 部 类 的 一 个 对 象 ， 否 则 不 可 能 创建 内 部 类 的 一 个 对 
象 。 这 是 由 于 内 部 类 的 对 象 已 同 创 建 它 的 外 部 类 的 对 象 “默默 ”地 连接 到 
然而 ， 如 果 生 成 一 个 static 内 部 类 ， 就 不 需要 指向 外 部 类 对 象 的 一 
MEJIA o 


7.6.6 从 内 部 类 继承 


由 于 内 部 类 构建 句 必 须 同 封装 类 对 象 的 一 个 句柄 联系 到 一 起 ， 所 以 从 一 
个 内 部 类 继承 的 时 候 ， 情 况 会 稍微 变 得 有 些 复杂 。 这 儿 的 问题 是 封装 类 
的 “秘密 ?向 柄 必须 获得 初始 化 ， 而 且 在 衍生 类 中 不 再 有 一 个 默认 的 对 象 
a 解决 这 个 问题 的 办 法 是 采用 一 种 特殊 的 语法 ， 明 确 建立 这 种 
关联 ， 


//: InheritInner.java 








// Inheriting an inner class 
class WithInner { 
class Inner {} 
} 
public class InheritInner 
extends WithInner.Inner { 
//! InheritInner() {} // Won't compile 
InheritInner(WithInner wi) { 
wi.super(); 
as 
public static void main(String[] args) { 
WithInner wi = new WithInner(); 
InheritInner ii = new InheritInner(wi); 
} 
} ///:~ 





从 中 可 以 看 到 ，InheritInner 只 对 内 部 类 进行 了 扩展 ， 没 有 扩展 外 部 类 。 
但 在 需要 创建 一 个 构建 器 的 时 候 ， 默 认 对 象 已 经 没有 和 意义， 我 们 不 能 只 
是 传递 封装 对 象 的 一 个 句柄 。 此 外 ， 必 须 在 构建 器 中 采用 下 述 语法 : 


enclosingClassHandle.super(); 

它 提 供 了 必要 的 句柄 ， 以 便 程序 正确 编译 。 

7.6.7 ALBA AY AAS aie ? 

若 创建 一 个 内 部 类 ， 然 后 从 封装 类 继承 ， 并 重新 定义 内 部 类 ， 那 么 会 出 
现 什么 情况 呢 ? 也 就 是 说 ， 我 们 有 可 能 履 盖 一 个 内 部 类 吗 ? 这 看 起 来 似 
乎 是 一 个 非常 有 用 的 概念 ， 但 “ 履 兰 ”一 个 内 部 闫 一 一 好 象 它 是 外 部 类 的 
男 一 个 方法 一 一 这 一 概念 实际 不 能 做 任何 事情 : 


//: BigEgg.java 











// An inner class cannot be overriden 
// like a method 
class Egg { 
protected class Yolk { 
public Yolk() { 


System.out.println("Egg.Yolk()"); 


} 


private Yolk y; 


public Egg() { 
System.out.println("New Egg()"); 


y = new Yolk(); 


} 


public class BigEgg extends Egg { 


public class Yolk { 
public Yolk() { 


System.out.println("BigEgg.Yolk()"); 


} 
public static void main(String[] args) { 
new BigEgg(); 
} 
} ///:~ 


默认 构建 器 是 由 编译 器 目 动 合成 的 ， 而 且 会 调用 基础 类 的 默认 构建 器 。 
大 家 或 许 会 认为 由 于 准备 创建 一 个 BigEgg， 上 以 会 使 用 Yolk 的 “被 履 
兰 ? 版 本 。 但 实际 情况 并 非 如 此 。 和 输出 如 下 : 


New Egg() 
Egg. Yolk() 


这 个 例子 简单 地 揭示 出 当 我 们 从 外 部 类 继承 的 时 候 ， 没 有 任何 额外 的 内 
部 类 继续 下 去 。 然 而 ， 仍 然 有 可 能 “明确 ”地 从 内 部 类 继承 : 


//: BigEgg2.java 


// Proper inheritance of an inner class 
class Egg2 { 
protected class Yolk { 
public Yolk() { 


System.out.println("Egg2.Yolk()"); 


} 
public void f() { 


System.out.println("Egg2.Yolk.f()"); 


} 


private Yolk y = new Yolk(); 
public Egg2() { 
System.out.println("New Egg2()"); 
} 
public void insertYolk(Yolk yy) { y = yy; } 
public void g() { y.f(); } 
} 
public class BigEgg2 extends Egg2 { 
public class Yolk extends Egg2.Yolk { 
public Yolk() { 
System.out.println("BigEgg2.Yolk()"); 
} 
public void f() { 


System.out.println("BigEgg2.Yolk.f()"); 


} 


public BigEgg2() { insertYolk(new Yolk()); } 


public static void main(String[] args) { 


Egg2 e2 = new BigEgg2(); 
e2.9(); 


} 
A f/f 
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法 insertYolk0O 人 允许 BigEgg2 将 它 自己 的 某 个 Yolk 对 象 上 调 造 型 至 Egg2 的 y 
句柄 。 上 所 以 当 gO 调 用 y.f0 的 时 候 ， 融 会 使 用 fO 被 履 凋 版 本 。 和 输出 结果 如 
下 : 

Egg2.Yolk() 

New Egg2() 

Egg2.Yolk() 

BigEgg2.Yolk() 

BigEgg2.Yolk.fQ) 


对 Egg2.YolkO 的 第 二 个 调用 是 BigEgg2.Yolk 构 建 器 的 基础 类 构建 器 调 
用 。 调 用 


g0 的 时 候 ， 可 发 现 使 用 的 是 {0 的 被 履 盖 版 本 。 
7.6.8 内 部 类 标识 和 从 


由 于 每 个 类 都 会 生成 一 个 .class 文 件 ， 用 于 容纳 与 如 何 创建 这 个 类 型 的 对 
象 有 关 的 所 有 信息 《这 种 信息 产生 了 一 个 名 为 Class 对 象 的 元 类 ) ， 所 以 
大 家 或 许 会 猪 到 内 部 类 也 必须 生成 相应 的 .class 文 件 ， 用 来 容纳 与 它们 的 
Class 对 象 有 关 的 信息 。 这 些 文件 或 类 的 名 字 遵 守 一 种 严格 的 形式 : 先是 
封装 类 的 名 字 ， 再 跟随 一 个 $， 再 跟随 内 部 类 的 名 字 。 例 如 ， 由 
InheritInner.java 创 建 的 .class 文 件 包括 : 








InheritInner.class 


WithInner$Inner.class 


WithInner.class 
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加 在 一 个 $ 以 及 外 部 类 标识 符 的 后 面 。 


这 种 生成 内 部 名 称 的 方法 除了 非常 简 蛙 和 和 直观 以 外 ， 也 非常 “健壮 ”， 可 
适应 大 多 数 场合 的 要 求 ( 注 释 @)) 。 由 于 它 是 Java 的 标准 命名 机 制 ， 所 
以 产生 的 文件 会 目 动 具备 “与 平台 无 关 ” 的 能 力 〈 注 意 Java 编 译 需 会 根据 
情况 改变 内 部 类 ， 使 其 在 不 同 的 平台 中 能 正常 工作 ) 。 


©: 但 在 另 一 方面 ， 由 于 “$” 也 是 Unix 外 壳 的 一 个 元 字符 ， 所 以 有 时 会 
在 列 出 .class 文 件 时 遇 到 麻烦 。 对 一 家 以 Unix 为 基础 的 公司 Sun 
来 说 ， 采 取 这 种 方案 显得 有 些 奇怪 。 我 的 猜测 是 他 们 根本 没有 仔细 考虑 
这 方面 的 问题 ， 而 是 认为 我 们 会 将 全 部 注意 力 上 自然 地 放 在 源码 文件 上 。 


7.6.9 为 什么 要 用 内 部 类 : 控制 框架 


到 目前 为 止 ， 大 家 已 接触 了 对 内 部 类 的 运作 进行 描述 的 大 量 语法 与 概 

念 。 但 这 些 并 不 能 真正 说 明 内 部 类 存在 的 原因 。 为 什么 Sun 要 如 此 麻烦 
地 在 Java 1.1 里 添加 这 样 的 一 种 基本 语言 特性 呢 ? 答案 就 在 于 我 们 在 这 
里 要 学 习 的 “控制 框架 ”。 


一 个 “应 用 程序 框架 * 是 指 一 个 或 一 系列 类 ， 它 们 专门 设计 用 来 解决 特定 
类 型 的 问题 。 为 应 用 应 用 程序 框架 ， 我 们 可 从 一 个 或 多 个 类 继承 ， 并 和 埠 
畜 其 中 的 部 分 方法 。 我 们 在 履 盖 方法 中 编写 的 代码 用 于 定制 由 那些 应 用 
程序 框架 提供 的 第 规 方案 ， 以 便 解 决 自己 的 实际 问题 。“ 控 制 框架 * 属 于 
应 用 程序 框架 的 一 种 特殊 类 型 ， 受 到 对 事件 啊 应 的 需要 的 文 配 ， 主 要 用 
来 响应 事件 的 一 个 系统 叫 作 “由 事件 驱动 的 系统 ”。 在 应 用 程序 设计 语言 
中 ， 最 重要 的 问题 之 一 便 是 “图 形 用 户 界 面 ”(GUI)〉， 它 几乎 完全 是 由 
事件 驱动 的 。 正 如 大 家 会 在 第 13 章 学 习 的 那样 ，Java 1.1 AWT 属 于 一 种 
控制 框架 ， 它 通过 内 部 类 完美 地 解决 了 GUI 的 问题 。 


为 理解 内 部 类 如 何 简 化 控制 框 染 的 创建 与 使 用 ， 可 认为 一 个 控制 框 染 的 
工作 就 是 在 事件 “ 右 绪 ”以 后 执行 它们 。 尺 管 “ 就 绪 ” 的 意思 很 多 ， 但 在 目 
前 这 种 情况 下 ， 我 们 却 是 以 计算 机 时 钟 为 基础 。 随 后 ， 请 认识 到 针对 控 





















































制 框架 需要 控制 的 东西 ， 框 架 内 并 未 包含 任何 特定 的 信息 。 首 先 ， 它 是 
一 个 特殊 的 接口 ， 描 述 了 所 有 控制 事件 。 它 可 以 是 一 个 抽象 类 ， 而 非 一 





//: Event.java 


// The common methods for any control event 
package c07.controller; 
abstract public class Event { 
private long evtTime; 
public Event(long eventTime) { 
evtTime = eventTime; 
} 
public boolean ready() { 
return System.currentTimeMillis() >= evtTime; 
} 
abstract public void action(); 
abstract public String description(); 


} ///:~ 


希望 Event (事件 ) 运行 的 时 候 ， 构 建 器 即 简单 地 捕获 时 间 。 同 时 
ready0 告 诉 我 们 何 时 该 运行 它 。 当 然 ，ready0 也 可 以 在 一 个 衍生 类 中 被 
覆盖 ， 将 事件 建立 在 除 时 间 以 外 的 其 他 东西 上 。 


action() 是 事件 就 绪 后 需要 调用 的 方法 ， 而 description() 提 供 了 与 事件 有 
关 的 文字 信息 。 


下 面 这 个 文件 包含 了 实际 的 控制 框架 ， 用 于 管理 和 触发 事件 。 第 一 个 类 
实际 只 是 一 个 “助手 ”类 ， 它 的 职责 是 容纳 Event 对 象 。 ee 

集合 蔡 换 它 。 而 且 通 过 第 8 章 的 学 习 ， 大 家 会 知道 另 一 些 集合 可 简化 我 
们 的 工作 ， 不 需要 我 们 编写 这 些 额 外 的 代码 : 


//: Controller.java 





// Along with Event, the generic 
// framework for all control systems: 
package c07.controller; 
// This is just a way to hold Event objects. 
class EventSet { 
private Event[] events = new Event[100]; 
private int index = 0; 
private int next = 0; 
public void add(Event e) { 
if(index >= events.length) 
return; // (In real life, throw exception) 
events[index++] = e; 
} 
public Event getNext() { 
boolean looped = false; 
int start = next; 
do { 


next = (next + 1) % events.length; 


// See if it has looped to the beginning: 
if(start == next) looped = true; 
// If it loops past start, the list 
// is empty: 
if((next == (start + 1) % events.length) 
&& looped) 
return null; 
} while(events[next] == null); 
return events[next]; 
} 
public void removeCurrent() { 


events[next] = null; 


} 
public class Controller { 
private EventSet es = new EventSet(); 
public void addEvent(Event c) { es.add(c); } 
public void run() { 
Event e; 
while((e = es.getNext()) != null) { 
if(e.ready()) { 
e.action(); 


System.out.println(e.description()); 


es.removeCurrent(); 


} 
} ///:~ 





EventSet 可 容纳 100 个 事件 《〈 知 在 这 里 使 用 来 目 第 8 章 的 一 个 “真实 ” 集 
合 ， 就 不 必 担 心 它 的 最 大 尺寸 ， 因 为 它 会 根据 情况 上 自动 改变 大 小 ) 。 
index〈 索 引 ) 在 这 里 用 于 跟踪 下 一 个 可 用 的 空间 ， 而 next (下 一 个 ) 大 
助 我 们 寻找 列表 中 的 下 一 个 事件 ， 了 解 上 自己 是 个 已 经 循环 到 头 。 在 对 
getNextO 的 调用 中 ， 这 一 点 是 至 关 重 要 的 ， 因 为 一 旦 运行 ，Event 对 象 就 
会 从 列表 中 删 去 (使 用 removeCurrent()) 。 所 以 getNext() 会 在 列表 中 问 
前 移动 时 遇 到 “空洞 ”。 


注意 removeCurrentO 并 不 只 是 指示 一 些 标志 ， 指 出 对 象 不 再 使 用 。 相 
反 ， 它 将 句柄 设 为 nall。 这 一 点 是 非常 重要 的 ， 因 为 假如 垃圾 收集 器 发 
现 一 个 句柄 仍 在 使 用 ， 歌 不 会 清除 对 象 。 知 认为 自己 的 句柄 可 能 象 现 在 
那么 最 好 将 其 设 为 null， 使 垃圾 收集 器 能 够 正常 地 清除 它 
[Te 














Controller 是 进行 实际 工作 的 地 方 。 它 用 一 个 EventSet 容 纳 自己 的 Event 对 
象 ， 而 且 addEvent() 允 许 我 们 向 这 个 列表 加 入 新 事件 。 但 最 重要 的 方法 
是 run()。 该 方法 会 在 EventSet 中 遍历， 搜索 一 个 准备 运行 的 Event 对 象 
ready()。 对 于 它 发 现 ready0 的 每 一 个 对 象 ， 都 会 调用 action() 方 法 ， 
打印 出 description0， 然 后 将 事件 从 列表 中 删 去 。 


注意 在 迄今 为 止 的 所 有 设计 中 ， 我 们 仍然 不 能 准确 地 知道 一 个 “事件 ”要 
做 什么 。 这 正 是 整个 设计 的 关键 ; 它 上 怎 样 “ 将 发 生变 化 的 东西 同 没 有 变 
化 的 东西 区 分 开 ”? 或 者 用 我 的 话 来 讲 , “改变 的 意图 ”造成 了 各 类 Event 
对 象 的 不 同行 动 。 我 们 通过 创建 不 同 的 Event 子 类 ， 从 而 表达 出 不 同 的 


人 一 一 


行动 。 
这 里 正 是 内 部 类 大 显 身 手 的 地 方 。 它 们 人 允许 我 们 做 两 件 事情 : 














(1) 在 单独 一 个 类 里 表达 一 个 控制 框架 应 用 的 全 部 实施 细 方 ， 从 而 完整 
地 封装 与 那个 实施 有 关 的 所 有 东西 。 内 部 类 用 于 表达 多 种 不 同类 型 的 
action()， 它 们 用 于 解决 实际 的 问题 。 除 此 以 外 ， 后 续 的 例子 使 用 了 
private 内 部 类 ， 所 以 实施 细 市 会 完全 隐藏 起 来 ， 可 以 安全 地 修改 。 


(2) ”内 部 类 使 我 们 具体 的 实施 变 得 更 加 巧妙 ， 因 为 能 方便 地 访问 外 部 类 
的 任何 成 员 。 知 不 具备 这 种 能 力 ， 代 码 看 起 来 就 可 能 没 那么 使 人 舒服 ， 
最 后 不 得 不 寻找 其 他 方法 解决 。 


现在 要 请 大 家 思考 控制 框架 的 一 种 具体 实施 方式 ， 它 设计 用 来 控制 温室 
(Greenhouse) 功能 (注释 4)) 。 每 个 行动 都 是 完全 不 同 的 : 控制 灯 
光 、 供 水 以 及 温度 目 动 调节 的 开 与 关 ， 控 制 啊 铃 ， 以 及 重新 启动 系统 。 
但 控制 框架 的 设计 宗旨 是 将 不 同 的 代码 方便 地 隔离 开 。 对 每 种 类 型 的 行 
都 要 继承 一 个 新 的 Event 内 部 类 ， 并 在 action0 内 编写 相应 的 控制 代 


O: 由 于 某 些 特殊 原因 ， 这 对 我 来 说 是 一 个 经 常 需要 解决 的 、 非 常 有 趣 
的 问题 ; 原来 的 例子 在 《C++ Inside & Out》 一 书 里 也 出 现 过 ， 但 Java 提 
供 了 一 种 更 令 人 舒适 的 解决 方案 。 


作为 应 用 程序 框架 的 一 种 典型 行为 ，GreenhouseControls 类 是 从 
Controller 继 承 的 : 























//: GreenhouseControls.java 


// This produces a specific application of the 
// control system, all in a single class. Inner 
// classes allow you to encapsulate different 
// functionality for each type of event. 
package c07.controller; 
public class GreenhouseControls 

extends Controller { 


private boolean light = false; 


private boolean water = false; 
private String thermostat = "Day"; 
private class LightOn extends Event { 
public LightOn(long eventTime) { 
super (eventTime) ; 
} 
public void action() { 
// Put hardware control code here to 
// physically turn on the light. 
light = true; 
} 
public String description() { 


return "Light is on"; 


j 


private class LightOff extends Event { 
public LightOff(long eventTime) { 
super (eventTime) ; 
} 
public void action() { 
// Put hardware control code here to 
// physically turn off the light. 


light = false; 


} 


public String description() { 


return "Light is off"; 


} 


private class WaterOn extends Event { 

public WaterOn(long eventTime) { 
super (eventTime) ; 

} 

public void action() { 
// Put hardware control code here 
water = true; 

} 

public String description() { 


return "Greenhouse water is on"; 


} 


private class Wateroff extends Event { 
public WaterOff(long eventTime) { 
super (eventTime) ; 
} 
public void action() { 


// Put hardware control code here 


water = false; 
} 
public String description() { 


return "Greenhouse water is off"; 


} 


private class ThermostatNight extends Event { 

public ThermostatNight(long eventTime) { 
super(eventTime) ; 

} 

public void action() { 
// Put hardware control code here 
thermostat = "Night"; 

} 

public String description() { 


return "Thermostat on night setting"; 


i 


private class ThermostatDay extends Event { 
public ThermostatDay(long eventTime) { 
super (eventTime); 


} 


public void action() { 


// Put hardware control code here 
thermostat = "Day"; 

} 

public String description() { 


return "Thermostat on day setting"; 


// An example of an action() that inserts a 
// new one of itself into the event list: 
private int rings; 

private class Bell extends Event { 

public Bell(long eventTime) { 
super (eventTime) ; 
} 
public void action() { 
// Ring bell every 2 seconds, rings times: 
System.out.println("Bing!"); 
if(--rings > 0) 
addEvent(new Bell( 
System.currentTimeMillis() + 2000)); 
} 
public String description() { 


return "Ring bell"; 


private class Restart extends Event { 
public Restart(long eventTime) { 
super(eventTime) ; 
} 
public void action() { 
long tm = System.currentTimeMillis(); 
// Instead of hard-wiring, you could parse 
// configuration information from a text 
// file here: 
rings = 5; 
addEvent(new ThermostatNight(tm) ); 
addEvent(new LightOn(tm + 1000)); 
addEvent(new LightOff(tm + 2000) ); 
addEvent(new WaterOn(tm + 3000)); 
addEvent(new WaterOff(tm + 8000)); 
addEvent(new Bell(tm + 9000)); 
addEvent(new ThermostatDay(tm + 10000)); 
// Can even add a Restart object! 
addEvent(new Restart(tm + 20000)); 


} 


public String description() { 


return "Restarting system"; 


} 
public static void main(String[] args) { 
GreenhouseControls gc = 
new GreenhouseControls(); 
long tm = System.currentTimeMillis(); 
gc.addEvent(gc.new Restart(tm)); 
gc.run(); 


} 
} ///:~ 


注意 light〈 灯 光 ) 、water〔 供 水 ) 、thermostat( 调 温 〉 以 及 rings 都 隶 
属于 外 部 类 GreenhouseControls， 所 以 内 部 类 可 以 毫 无 阻碍 地 访问 那些 
字段 。 此 外 ， 大 多 数 action(0) 方 法 也 涉及 到 某 些 形式 的 硬件 控制 ， 这 通常 
都 要 求 发 出 对 非 Java 代 码 的 调用 。 


大 多 数 Event 类 看 起 来 都 是 相似 的 ， 但 Bell (44) 和 Restart (重启 〉 属 于 
特殊 情况 。Bell 会 发 出 啊 声 ， 知 尚未 啊 铃 足够 的 次 数 ， 它 会 在 事件 列表 
里 添加 一 个 新 的 Bell 对 象 ， 所 以 以 后 会 再 度 啊 铃 。 请 注意 内 部 类 看 起 来 
为 什么 总 是 类 似 于 多 重 继承 : Bell 拥 有 Event 的 所 有 方法 ， 而 且 也 拥有 外 
部 类 GreenhouseControls 的 所 有 方法 。 


Restart 负 责 对 系统 进行 初始 化 ， 所 以 会 添加 所 有 必要 的 事件 。 当 然 ， 一 
种 更 灵活 的 做 法 是 避免 进行 “ 硬 编码 ”， 而 是 从 一 个 文件 里 读 入 它们 《第 
10 章 的 一 个 练习 会 要 求 大 家 修改 这 个 例子 ， 从 而 达到 这 个 目标 ) 。 由 于 
RestartO 仅 仅 是 另 一 个 Event 对 象 ， 所 以 也 可 以 在 Restartaction(0) 里 添加 一 
个 Restart 对 象 ， 使 系统 能 够 定期 重启 。 在 main(0) 中 ， 我 们 需要 做 的 全 部 
事情 就 是 创建 一 个 GreenhouseControls 对 象 ， 并 添加 一 个 Restart 对 象 ， 令 
FUT ERK. 











这 个 例子 应 该 使 大 家 对 内 部 类 的 价值 有 一 个 更 加 深刻 的 认识 ， 特 别 是 在 
一 个 控制 框架 里 使 用 它们 的 时 候 。 些 外， 在 第 13 章 的 后 半 部 分 ， 大 家 还 
会 看 到 如 何 巧妙 地 利用 内 部 类 描述 一 个 图 形 用 户 界面 的 行为 。 完 成 那里 
的 学 习 后 ， 对 内 部 类 的 认识 将 上 升 到 一 个 前 所 未 有 的 新 高 度 。 











7.7 构建 部 和 多 形 性 


同 往常 一 样 ， 构 建 器 与 其 他 种 类 的 方法 是 有 区 别 的 。 在 涉及 到 多 形 性 的 
问题 后 ， 这 种 方法 依然 成 立 。 尽 管 构 建 器 并 不 具有 多 形 性 《“ 即 便 可 以 使 
用 一 种 “虚拟 构建 融 ' 一 一 将 在 第 11 章 介绍 ) ， 但 仍然 非常 有 必要 理解 构 
建 锅 如 何在 复杂 的 分 级 结构 中 以 及 随同 多 形 性 使 用 。 这 一 理解 将 有 助 于 
大 家 避免 陷入 一 些 令 人 不 快 的 纠纷 。 


7.7.1 构建 器 的 调用 顺序 


构建 堪 调 用 的 顺序 已 在 第 4 章 进 行 了 简要 次 明 ， 但 那 是 在 继承 和 多 形 性 
问题 引入 之 前 说 的 话 。 


用 于 基础 类 的 构建 磺 肯 定 在 一 个 衍生 类 的 构建 句 中 调用 ， 而 且 逐 渐 问 上 
链接 ， 使 每 个 基础 类 使 用 的 构建 侣 都 能 得 到 调用 。 之 所 以 要 这 样 做 ， 是 
由 于 构建 器 负 有 一 项 特殊 任务 ， 检查 对 象 是 否 得 到 了 正确 的 构建 。 一 个 
衍生 类 只 能 访问 它 自己 的 成 员 ， 不 能 访问 基础 类 的 成 员 〈( 这 些 成 员 通 委 
都 具有 private 属 性 ) 。 只 有 基础 类 的 构建 器 在 初始 化 目 己 的 元 素 时 才 知 
道 正 确 的 方法 以 及 拥有 适当 的 权限 。 所 以 ， 必 须 令 所 有 构建 器 都 得 到 调 

， 否 则 整个 对 象 的 构建 就 可 能 不 正确 。 那 正 是 编译 器 为 什么 要 强迫 对 
衍生 类 的 每 个 部 分 进行 构建 器 调用 的 原因 。 在 衍生 类 的 构建 器 主体 中 ， 
右 我 们 没有 明确 指定 对 一 个 基础 类 构建 豆 的 调用 ， 它 就 会 < 默默 ?地 调用 
默认 构建 器 。 如 果 不 存 在 默认 构建 问 ， 编 译 圳 就 会 报告 一 个 错误 《〈 重 茶 
个 类 没有 构建 问 ， 编 译 需 会 目 动 组 织 一 个 默认 构建 问 ) 。 


下 面 让 我 们 看 看 一 个 例子 ， 它 展示 了 按 构 建 顺序 进行 合成 、 继 承 以 及 多 
形 性 的 效果 : 


//: Sandwich,java 
































// Order of constructor calls 
class Meal { 


Meal() { System.out.printin("Meal()"); } 


class Bread { 
Bread() { System.out.println("Bread()"); } 
} 
class Cheese { 
Cheese() { System.out.println("Cheese()"); } 
} 
class Lettuce { 
Lettuce() { System.out.println("Lettuce()"); } 
} 
class Lunch extends Meal { 
Lunch() { System.out.println("Lunch()");} 
} 
class PortableLunch extends Lunch { 
PortableLunch() { 


System.out.println("PortableLunch()"); 


} 


class Sandwich extends PortableLunch { 
Bread b = new Bread(); 
Cheese c = new Cheese(); 
Lettuce 1 = new Lettuce(); 
Sandwich() { 


System. out.println("Sandwich()"); 


} 


public static void main(String[] args) { 
new Sandwich(); 


} 
} ///:~ 


这 个 例子 在 其 他 类 的 外 部 创建 了 一 个 复杂 的 类 ， 而 且 每 个 类 都 有 一 个 构 
建 右 对 自己 进行 了 宣布 。 其 中 最 重要 的 类 是 Sandwich， 它 反映 出 了 三 个 
级 别 的 继承 〈 知 将 从 Object 的 默认 继承 算 在 内 ， 就 是 四 级 ) 以 及 三 个 成 
员 对 象 。 在 main() 里 创建 了 一 个 Sandwich 对 象 后 ， 输 出 结果 如 下 : 


Meal() 





Lunch() 
PortableLunch( ) 
Bread ( ) 
Cheese() 
Lettuce() 


Sandwich( ) 





IO AT FART RRS A Dad FB TT EIU : 


(1) 调用 基础 类 构建 器 。 这 个 步骤 会 不 断 重 复 下 去 ， 首 移 得 到 构建 的 是 
人 
Ro 


(2) 按 声明 顺序 调用 成 员 初 始 化 模块 。 
(3) 调用 衍生 构建 器 的 主体 。 








构建 器 调用 的 顺序 是 非常 重要 的 。 进 行 继 承 时 ， 我 们 知道 关于 基础 类 的 
一 切 ， 并 且 能 访问 基础 类 的 任何 public 和 protected 成 员 。 这 意味 着 当 我 
们 在 衍生 类 的 时 候 ， 必 须 能 假定 基础 类 的 所 有 成 员 都 是 有 效 的 。 采 用 一 
种 标准 方法 ， 构 建行 动 已 经 进行 ， 所 以 对 象 所 有 部 分 的 成 员 均 已 得 到 构 
建 。 但 在 构建 器 内 部 ， 必 须 保 证 使 用 的 所 有 成 员 都 已 构建 。 为 达到 这 个 
要 求 ， 唯 一 的 办 法 就 是 首先 调用 基础 类 构建 问 。 然 后 在 进入 衍生 类 构建 
器 以 后 ， 我 们 在 基础 类 能 够 访问 的 所 有 成 员 都 已 得 到 初始 化 。 此 外 ， 所 
有 成 员 对 象 〈 亦 即 通过 合成 方法 置 于 类 内 的 对 象 ) 在 类 内 进行 定义 的 时 
候 〈 比 如 上 例 中 的 b，c 和 1) ， 由 于 我 们 应 尽 可 能 地 对 它们 进行 初始 
化 ， 所 以 也 应 保证 构建 器 内 部 的 所 有 成 员 均 为 有 效 。 徊 坚持 按 这 一 规则 
行事 ， 会 有 助 于 我 们 确定 所 有 基础 类 成 员 以 及 当前 对 象 的 成 员 对 象 均 已 
获得 正确 的 初始 化 。 但 不 笠 的 是 ， 这 种 做 法 并 不 适用 于 所 有 情况 ， 这 将 
在 下 一 市 具体 说 明 。 


7.7.2 继承 和 finalize() 


通过 “合成 ”方法 创建 新 类 时 ， 永 远 不 必 担 心 对 那个 类 的 成 员 对 象 的 收尾 
工作 。 每 个 成 员 都 是 一 个 独立 的 对 象 ， 所 以 会 得 到 正常 的 垃圾 收集 以 及 
收尾 处 理 无 论 它 是 不 是 不 自己 某 个 类 一 个 成 员 。 但 在 进行 初始 化 的 
时 候 ， 必 须 覆 盖 衍 生 类 中 的 finalize() 方 法 一 一 如 果 已 经 设计 了 某 个 特殊 
的 清除 进程 ， 要 求 它 必 须 作为 垃圾 收集 的 一 部 分 进行 。 才 兰 衍生 类 的 

finalizeO0 时 ， 务 必 记 住 调 用 finalize0 的 基础 类 版 本 。 人 和 否则， 基础 类 的 初 
始 化 根本 不 会 有 发生。 下 面 这 个 例子 便 是 明证 : 


//: Frog.java 



































// Testing finalize with inheritance 
class DoBaseFinalization { 

public static boolean flag = false; 
} 
class Characteristic { 

String s; 


Characteristic(String c) { 


S = C} 
System.out.println( 
"Creating Characteristic " + s); 
} 
protected void finalize() { 
System.out.printin( 


"finalizing Characteristic " + s); 


J 


class LivingCreature { 
Characteristic p = 
new Characteristic("is alive"); 
LivingCreature() { 
System.out.println("LivingCreature()"); 
} 
protected void finalize() { 
System.out.printin( 
"LivingCreature finalize"); 
// Call base-class version LAST! 
if (DoBaseFinalization.flag) 
try { 
super.finalize(); 


} catch(Throwable t) {} 


} 


class Animal extends LivingCreature { 
Characteristic p = 
new Characteristic("has heart"); 
Animal() { 
System.out.println("Animal()"); 
} 
protected void finalize() { 
System.out.println("Animal finalize"); 
if (DoBaseFinalization. flag) 
try { 
super.finalize(); 


} catch(Throwable t) {} 


} 


class Amphibian extends Animal { 
characteristic p = 
new Characteristic("can live in water"); 
Amphibian() { 
System.out.println("Amphibian()"); 


} 


protected void finalize() { 


System.out.printin("Amphibian finalize"); 
if (DoBaseFinalization.flag) 
try { 
super .finalize(); 


} catch(Throwable t) {} 


} 


public class Frog extends Amphibian { 


Frog() { 


System.out.println("Frog()"); 
} 
protected void finalize() { 
System.out.println("Frog finalize"); 
if (DoBaseFinalization.flag) 
try { 
super.finalize(); 
} catch(Throwable t) {} 
} 
public static void main(String[] args) { 
if(args.length != 0 && 
args[0].equals("finalize" ) ) 
DoBaseFinalization.flag = true; 


else 


System.out,println("not finalizing bases"); 
new Frog(); // Instantly becomes garbage 
System.out.printlin("bye!"); 

// Must do this to guarantee that all 
// finalizers will be called: 
System. runFinalizersOnExit(true); 


} 
SA 


DoBasefinalization 类 只 是 简单 地 容纳 了 一 个 标志 ， 问 分 级 结构 中 的 每 个 
类 指出 是 否 应 调用 super.finalize()。 这 个 标志 的 设置 建立 在 命令 行 参 数 的 
基础 上 ， 所 以 能 够 在 进行 和 不 进行 基础 类 收尾 工作 的 前 提 下 查看 行为 。 


分 级 结构 中 的 每 个 类 也 包含 了 Characteristic 类 的 一 个 成 员 对 象 。 大 家 可 
以 看 到 ， 无 论 是 否 调用 了 基础 类 收尾 模块 ，Characteristic 成 员 对 象 都 肯 
定 会 得 到 收尾 (清除) 处 理 。 


每 个 被 覆盖 的 finalize0 至 少 要 拥有 对 protected 成 员 的 访问 权力 ， 因 为 
Object 类 中 的 finalize0) 方 法 具有 protected 属 性 ， 而 编译 器 不 允许 我 们 在 继 
承 过 程 中 消除 访问 权限 (“友好 的 ” 比 “ 受 到 保护 的 "具有 更 小 的 访问 权 
限 ) 。 


在 Frog.main0 中 ，DoBaseFinalization 标 志 会 得 到 配置 ， 而 且 会 创建 单独 
一 个 Frog 对 象 。 请 记 住 垃圾 收集 (特别 是 收尾 工作 〉 可 能 不 会 针对 任何 
特定 的 对 象 改 生 ， 所 以 为 了 强制 采取 这 一 行动 ， 
System.runFinalizersOnExit(true) 这 加 了 额外 的 开销 ， 以 保证 收尾 工作 的 
正常 进行 。 寿 没有 基础 类 初始 化 ， 则 输出 结果 是 : 


not finalizing bases 














Creating Characteristic is alive 


LivingCreature( ) 

Creating Characteristic has heart 
Animal ( ) 

Creating Characteristic can live in water 
Amphibian( ) 

Frog() 

bye! 

Frog finalize 

finalizing Characteristic is alive 
finalizing Characteristic has heart 


finalizing Characteristic can live in water 


从 中 可 以 看 出 确实 没有 为 基础 类 Frog 调 用 收尾 模块 。 但 假如 在 命令 行 加 
入 “finalize” 自 变量 ， 则 会 获得 下 述 结果 : 


Creating Characteristic is alive 


LivingCreature() 

Creating Characteristic has heart 
Animal( ) 

Creating Characteristic can live in water 
Amphibian( ) 

Frog() 

bye! 


Frog finalize 


Amphibian finalize 

Animal finalize 

LivingCreature finalize 

finalizing Characteristic is alive 
finalizing Characteristic has heart 


finalizing Characteristic can live in water 








尽管 成 员 对 象 按 照 与 它们 创建 时 相同 的 顺序 进行 收尾 ， 但 从 技术 角度 
说 ， 并 没有 指定 对 象 收尾 的 顺序 。 但 对 于 基础 类 ， 我 们 可 对 收尾 的 顺序 
进行 控制 。 采 用 的 最 佳 顺 序 正 是 在 这 里 采用 的 顺序 ， 它 与 初始 化 顺序 正 
好 相反 。 按 照 与 C++ 中 用 于 “破坏 器 ?相同 的 形式 ， 我 们 应 该 首先 执行 衍 
生 类 的 收尾 ， 再 是 基础 类 的 收尾 。 这 是 由 于 衍生 类 的 收尾 可 能 调用 基础 
类 中 相同 的 方法 ， 要 求 基础 类 组 件 仍然 处 于 活动 状态 。 因 此 ， 必 须 提前 
将 它们 清除 《破坏 ) 。 


7.7.3 构建 器 内 部 的 多 形 性 方法 的 行为 


构建 器 调用 的 分 级 结构 《顺序 ) 为 我 们 带 来 了 一 个 有 趣 的 问题 ， 或 者 说 
让 我 们 进入 了 一 种 进退 两 难 的 局 面 。 知 当前 位 于 一 个 构建 器 的 内 部 ， 同 
时 调用 准备 构建 的 那个 对 象 的 一 个 动态 绑 定 方法 ， 那 么 会 出 现 什 么 情况 
Ne? 在 原始 的 方法 内 部 ， 我 们 完全 可 以 想象 会 及 生 什么 一 一 动态 绑 定 的 
调用 会 在 运行 期 间 进 行 解析 ， 因 为 对 象 不 知道 它 到 底 从 属于 方法 所 在 的 
那个 类 ， 还 是 从 属于 从 它 衍 生出 来 的 茶 些 类 。 为 保持 一 致 性 ， 大 家 也 许 
会 认为 这 应 该 在 构建 器 内 部 发 生 。 


但 实际 情况 并 非 完 全 如 此 。 知 调用 构建 右 内 部 一 个 动态 绑 定 的 方法 ， 会 
使 用 那个 方法 和 被 履 盖 的 定义 。 然 而 ， 产 生 的 效果 可 能 并 不 如 我 们 所 愿 ， 
而 且 可 能 造成 一 些 难于 发 现 的 程序 错误 。 


从 概念 上 讲 ， 构 建 器 的 职责 是 让 对 象 实际 进入 存在 状态 。 在 任何 构建 器 
内 部 ， 整 个 对 象 可 能 只 是 得 到 部 分 组 织 一 一 我 们 只 知道 基础 类 对 象 已 得 
到 初始 化 ， 但 却 不 知道 哪些 类 已 经 继承 。 然 而 ， 一 个 动态 绑 定 的 方法 调 
用 却 会 在 分 级 结构 里 “ 疝 前 ”或 者 “ 回 外 ”前 进 。 它 调用 位 于 衍生 类 里 的 一 
个 方法 。 如 果 在 构建 各 内 部 做 这 件 事情 ， 那 么 对 于 调用 的 方法 ， 它 要 操 









































纵 的 成 员 可 能 尚未 得 到 正确 的 初始 化 一 一 这 显 
通过 观察 下 面 这 个 例子 ， 这 个 问题 便 会 昭然 


//: PolyConstructors.java 


// Constructors and polymorphism 


然 不 是 我 们 所 希望 的 。 


AAG Ha: 


// don't produce what you might expect. 


abstract class Glyph { 
abstract void draw(); 
Glyph() { 
System.out.println("Glyph() before 


draw(); 


draw()"); 


System.out.printin("Glyph() after draw()"); 


J 


class RoundGlyph extends Glyph { 
int radius = 1; 
RoundGlyph(int r) { 
radius = r; 
System.out.printin( 
"RoundGlyph.RoundGlyph(), radius 
+ radius); 


} 


void draw() { 


System.out .printlLn( 


"RoundGlyph.draw(), radius = " + radius); 


} 
public class PolyConstructors { 
public static void main(String[] args) { 
new RoundGlyph(5); 
} 
} ///:~ 


在 Glyph 中 ，draw() 方 法 是 “抽象 的 ”(abstract) ， 所 以 它 可 以 被 其 他 方法 
敌 产 。 事 实 上 ， 我 们 在 RoundGlyph 中 不 得 不 对 其 进行 禾 新 。 但 Glyph 构 

建 句 会 调用 这 个 方法 ， 而 且 调 用 会 在 RoundGlyph.draw0O 中 止 ， 这 看 起 来 
似乎 是 有 意 的 。 但 请 看 看 输出 结果 : 


Glyph() before draw() 


RoundGlyph.draw(), radius = 0 
Glyph() after draw() 


RoundGlyph.RoundGlyph(), radius = 5 


当 Glyph 的 构建 器 调用 drawO 时 ，radius 的 值 甚至 不 是 默认 的 初始 值 1， 而 
是 0。 这 可 能 是 由 于 一 个 点 号 或 者 屏幕 上 根本 什么 都 没有 男 而 造成 的 。 
这 样 就 不 得 不 开始 查找 程序 中 的 错误 ， 试 着 找 出 程序 不 能 工作 的 原因 。 


前 一 节 讲 述 的 初始 化 顺序 并 不 十 分 完整 ， 而 那 古 解决 问题 的 关键 所 在 。 
初始 化 的 实际 过 程 是 这 样 的 : 


(1) 在 采取 其 他 任何 操作 之 前 ， 为 对 象 分 配 的 存储 空间 初始 化 成 二 进 制 
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(2) 就 象 前 面 叙 述 的 那样 ， 调 用 基础 类 构建 器 。 此 时 ， 补 和 履 盖 的 draw0) 方 
法 会 得 到 调用 〈 的 确 是 在 RoundGlyph 构 建 器 调用 之 前 ) ， 此 时 会 发 现 
radius 的 值 为 0， 这 是 由 于 步骤 (1) 造 成 的 。 


(3) 按照 原先 声明 的 顺序 调用 成 员 初 始 化 代码 。 
(4) 调用 衍生 类 构建 器 的 主体 。 


采取 这 些 操作 要 求 有 一 个 前 提 ， 那 就 是 所 有 东西 都 至 少 要 初始 化 成 零 
《或 者 东 些 特殊 数据 类 型 与 “ 零 ” 等 价 的 值 ) ， 而 不 是 仅仅 留 作 垃圾 。 其 
中 包括 通过 “合成 ?技术 藤 入 一 个 类 内 部 的 对 象 句 柄 。 如 果 假 在 二 记 初 始 
化 那个 句柄 ， 就 会 在 运行 期 间 出 现 违例 事件 。 其 他 所 有 东西 都 会 变 成 
零 ， 这 在 观看 结果 时 通常 是 一 个 严重 的 警告 信号 。 


在 另 一 方面 ， 应 对 这 个 程序 的 结果 提高 警惕 。 从 逻辑 的 角度 说 ， 我 们 似 
乎 已 进行 了 无 懈 可 击 的 设计 ， 所 以 它 的 错误 行为 令 人 非常 不 可 思议 。 而 
且 没 有 从 编译 器 那里 收 到 任何 报错 信息 《C++ 在 这 种 情况 下 会 表现 出 更 
| 
间 才 能 找 出 。 


因此 ， 设 计 构 建 器 时 一 个 特别 有 效 的 规则 是 : 用 尽 可 能 简单 的 方法 使 对 
象 进 入 就 绪 状 态 : 如 果 可 能 ， 避 人 免 调 用 任何 方法 。 在 构建 器 内 唯一 能 够 
安全 调用 的 是 在 基础 类 中 具有 final 属 性 的 那些 方法 (也 适用 于 private 方 
法 ， 它 们 目 动 具有 final 属 性 ) 。 这 些 方法 不 能 被 覆盖 ， 所 以 不 会 出 现 上 
述 潜在 的 问题 。 




















7.8 通过 继承 进行 设计 


学 习 了 多 形 性 的 知识 后 ， 由 于 多 形 性 是 如 此 “聪明 ”的 一 种 工具 ， 所 以 看 
起 来 似乎 所 有 东西 都 应 该 继承 。 但 假如 过 度 使 用 继承 技术 ， 也 会 使 自己 
的 设计 变 得 不 必要 地 复杂 起 来 。 事 实 上 ， 当 我 们 以 一 个 现成 类 为 基础 建 
一 个 新 类 时 ， 如 首先 选择 继承 ， 会 使 情况 变 得 异常 复杂 。 

一 个 更 好 的 思路 是 首先 选择 “合成 ”一 一 如 果 不 能 十 分 确定 自己 应 使 用 哪 
= 合成 不 会 强迫 我 们 的 程序 设计 进入 继承 的 分 级 HAR. B, £ 


成 显得 更 加 灵 范 ， 因为 可 以 动态 选择 一 种 类 型 (以 及 行为 ) ， 而 继承 要 
求 在 编译 期 间 准 确 地 知道 一 种 类 型 。 下 面 这 个 例子 对 此 进行 了 阐释 : 


//: Transmogrify.java 











// Dynamically changing the behavior of 
// an object via composition. 
interface Actor { 
void act(); 
} 
class HappyActor implements Actor { 
public void act() { 


System.out.println("HappyActor") ; 


} 
class SadActor implements Actor { 
public void act() { 


System.out.println("SadActor"); 


} 
class Stage { 
Actor a = new HappyActor(); 
void change() { a = new SadActor(); } 
void go() { a.act(); } 
} 
public class Transmogrify { 
public static void main(String[] args) { 
Stage s = new Stage(); 
s.go(); // Prints "HappyActor" 
s.change(); 
s.go(); // Prints "SadActor" 
} 


E E 


在 这 里 ， 一 个 Stage 对 象 包 含 了 指向 一 个 Actor 的 句 顶 ， 后 者 被 初始 化 成 
一 个 HappyActor 对 象 。 这 意味 着 go() 会 产生 特定 的 行为 。 但 由 于 句柄 在 
运行 期 间 可 以 重新 与 一 个 不 同 的 对 象 绑 定 或 结合 起 来 ， 所 以 SadActor 对 
象 的 句柄 可 在 a 中 得 到 替换 ， 然后 由 go0 产 生 的 行为 发 生 改 变 。 这 样 一 
来 ， 我 们 在 运行 期 间 就 获得 了 很 大 的 灵活 性 。 与 此 相反 ， 我 们 不 能 在 运 
行 期 间 换 用 不 同 的 形式 来 进行 继承 ; 它 要 求 在 编译 期 间 完 全 决定 下 来 。 


一 条 常规 的 设计 准则 是 : 用 继承 表达 行为 间 的 差异 ， 并 用 成 员 变 量 表达 
状态 的 变化 。 在 上 述 例子 中 ， 两 者 都 得 到 了 应 用 : 继承 了 两 个 不 同 的 
类 ， 用 于 表达 act(0) 方 法 的 差异 ， 而 Stage 通 过 合成 技术 允许 它 自己 的 状态 
发 生变 化 。 在 这 种 情况 下 ， 那 种 状态 的 改变 同时 也 产生 了 行为 的 变化 。 























7.8.1 纯 继 承 与 扩展 


学 习 继 承 时 ， 为 了 创建 继承 分 级 结构 ， 看 来 最 明显 的 方法 是 采取 一 
种 “纯粹 ”的 手段 。 也 就 是 说 ， 只 有 在 基础 类 或 “接口 ”中 已 建立 的 方法 才 
HY EATS PM m OP KA: 















erase() 
可 将 其 描述 成 一 种 纯粹 的 “属于 ”关系 ， 因 为 一 个 类 的 接口 已 规定 了 它 到 
底 “ 是 什么 ”或 者 “属于 什么 ”。 通 过 继承 ， 可 保证 所 有 衍生 类 都 只 拥有 基 


础 类 的 接口 。 如 果 按 上 述 示意 图 操作 ， 和 衍生 出 来 的 类 除了 基础 类 的 接口 
之 外 ， 也 不 会 再 拥有 其 他 什么 。 


可 将 其 想象 成 一 种 “ 纯 符 换 ”， 因 为 衍生 类 对 象 可 为 基础 类 完美 地 蔡 换 
人 
Co DF 下 JR: 


Talks to Shape -0 Circle, Square, 
- - Message Line, or new type 
Is-a of Shape 
relationship 


也 就 是 说 ， 基 础 类 可 接收 我 们 友 给 衍生 类 的 任何 消 恩 ， 因 为 两 者 拥有 完 
全 一 致 的 接口 。 我 们 要 做 的 全 部 事情 残 是 从 衍生 上 济 造 型 ， 而 且 永 远 不 
再 要 回 过 头 来 检查 对 象 的 准确 类 型 是 什么 。 所 有 细节 都 已 通过 多 形 性 获 
得 了 完美 的 控制 。 


若 按 这 种 思路 考虑 问题 ， 那 么 一 个 纯粹 的 “属于 ”关系 似乎 是 唯一 明智 的 
设计 方法 ， 其 他 任何 设计 方法 都 会 叶 致 混乱 不 清 的 思路 ， 而 且 在 定义 上 
存在 很 大 的 困难 。 但 这 种 想法 又 属于 为 一 个 极 并 。 经 过 细致 的 研究 ， 我 











们 发 现 扩 展 接口 对 于 一 些 特 定 问 题 来 说 是 特别 有 效 的 方案 。 可 将 其 称 
为 “类 似 于 ”关系 ， 因 为 扩展 后 的 衍生 类 “类 似 于 ?基础 类 一 一 它们 有 相同 
的 基础 接口 一 一 但 它 增 加 了 一 些 特性 ， 要 求 用 额外 的 方法 加 以 实现 。 如 
BATA: 


void fe) 
void gt} 
a 


void f) 
void gt} 
void uć) 


void w) 
void wt} 

















Assume this 
k represents a big 
interface 


"Is-like-a" 


Extending 
the interface 







尽管 这 是 一 种 有 用 和 明智 的 做 法 〈 由 具体 的 环境 决定 ) ， 但 它 也 有 一 个 
缺点 : 衍生 类 中 对 接口 扩展 的 那 一 部 分 不 可 在 基础 类 中 使 用 。 所 以 一 旦 
上 漳 造 型 ， 就 不 可 再 调用 新 方法 : 








各 在 此 时 不 进行 上 漳 造 型 ， 则 不 会 出 现 此 类 问题 。 但 在 许多 情况 下 ， 都 
需要 重新 核实 对 象 的 准确 类 型 ， 使 自己 能 访问 那个 类 型 的 扩展 方法 。 在 
后 面 的 小 节 里 ， 我 们 具体 讲述 了 这 是 如 何 实现 的 。 


7.8.2 下 淹 造 型 与 运行 期 类 型 标识 


由 于 我 们 在 上 调 造 型 《在 继承 结构 中 癌 上 移动 ) 期 间 丢 失 了 有 具体 的 类 型 
信息 ， 所 以 为 了 获取 具体 的 类 型 信息 一 一 亦 即 在 分 级 结构 中 疝 下 移动 

我 们 必须 使 用 “下 调 造 型 >? 技术。 然而 ， 我 们 知道 一 个 上 漳 造 型 肯定 
是 安全 的 ; 基础 类 不 可 能 再 拥有 一 个 比 衍 生 类 更 大 的 接口 。 因 此 ， 我 们 
通过 基础 类 接口 发 送 的 每 一 条 消息 都 衣 定 能 够 接收 到 。 但 在 进行 下 调 造 




















型 的 时 候 ， 我 们 〈 举 个 例子 来 说 ) 并 不 真 的 知道 一 个 几何 形状 实际 是 一 
个 圆 ， 它 完全 可 能 是 一 个 三 角形 、 方 形 或 者 其 他 形状 。 


void f 
void g6 
只 


/ 
i [void f 

: |void gt) 
void ut) 
void vO 
void w) 









Downcast: 


Upcast: 
always must be 
safe checked 








为 解决 这 个 问题 ， 必 须 有 一 种 办 法 能 够 保证 下 漳 造 型 正确 进行 。 只 有 这 
样 ， 我 们 才 不 会 冒 然 造型 成 一 种 错误 的 类 型 ， 然 后 发 出 一 条 对 象 不 可 能 
收 到 的 消息 。 这 样 做 是 非 第 不 安全 的 。 


在 某 些 语言 中 《如 C++) ， 为 了 进行 保证 “类 型 安全 ”的 下 渊 造型 ， 必 须 
采取 特殊 的 操作 。 但 在 Java 中 ， 所 有 造型 都 会 自动 得 到 检查 和 核实 ! 所 
以 即使 我 们 只 是 进行 一 次 普通 的 括 弧 造型 ， 进 入 运行 期 以 后 ， 仍 然 会 旱 
无 留情 地 对 这 个 造型 进行 检查 ， 保 证 它 的确 是 我 们 希望 的 那 种 类 型 。 如 
果 不 是 ， 就 会 得 到 一 个 ClassCastException (类 造型 违例 ) 。 在 运行 期 间 
对 类 型 进行 检查 的 行为 叫 作 “运行 期 类 型 标识 ”(RTTI〉。 下 面 这 个 例子 
向 大 家 演示 了 RTITI 的 行为 : 


//: RTTI.java 








// Downcasting & Run-Time Type 
// Identification (RTTI) 
import java.util.*; 

class Useful { 


public void f() {} 


public void g() {} 
} 
class MoreUseful extends Useful { 
public void f() {} 
public void g() {} 
public void u() {} 
public void v() {} 
public void w() {} 
} 
public class RTTI { 
public static void main(String[] args) { 
Useful[] x = { 
new Useful(), 
new MoreUseful( ) 
}; 
x[0].f(); 
x[1].9(); 
// Compile-time: method not found in Useful: 
//\ x[1].u(); 
((MoreUseful)x[1]).u(); // Downcast/RTTI 


((MoreUseful)x[0]).u(); // Exception thrown 


de 


和 在 示意 图 中 一 样 ，MoreUseful (更 有 用 的 ) 对 Useful (有 用 的 ) 的 接 
口 进行 了 扩展 。 但 由 于 它 是 继承 来 的 ， 所 以 也 能 上 漳 造 型 到 一 个 
Useful。 我 们 可 看 到 这 会 在 对 数组 x〈 位 于 main0 中 ) 进行 初始 化 的 时 候 
发 生 。 由 于 数组 中 的 两 个 对 象 都 属于 Useful 类 ， 所 以 可 将 f0 和 8g0O 方 法 同 
时 发 给 它们 两 个 。 而 且 假 如 试图 调用 u0 〈 它 只 存在 于 MoreUseful) , iit 
会 收 到 一 条 编译 期 出 错 提示 。 


若 想 访问 一 个 MoreUseful 对 象 的 扩展 接口 ， 可 试 着 进行 下 济 造 型 。 如 果 
它 是 正确 的 类 型 ， 这 一 行动 就 会 成 功 。 否 则 ， 就 会 得 到 一 个 
ClassCastException。 我 们 不 必 为 这 个 违例 编写 任何 特殊 的 代码 ， 因 为 它 
指出 的 是 一 个 可 能 在 程序 中 任何 地 方 发 生 的 一 个 编程 错误 。 


RTTI 的 意义 远 不 仅仅 反映 在 造型 处 理 上 。 例 如 ， 在 试图 下 淹 造 型 之 
前 ， 可 通过 一 种 方法 了 解 自己 处 理 的 是 什么 类 型 。 整 个 第 11 章 都 在 讲述 
Java 运 行 期 类 型 标识 的 方方面面 。 














7.9 总 结 


“多 形 性 ?意味 着 “不 同 的 形式 ?。 在 面 问 对 象 的 程序 设计 中 ， 我 们 有 相同 
的 外 观 《〈 基 础 类 的 通用 接口 ) 以 及 使 用 那个 外 观 的 不 同形 式 : 动态 绑 定 
或 组 织 的 、 不 同 版 本 的 方法 。 


通过 这 一 章 的 学 习 ， 大 家 已 知道 假如 不 利用 数据 抽象 以 及 继承 技术 ， 惑 
不 可 能 理解 、 甚 至 去 创建 多 形 性 的 一 个 例子 。 多 形 性 是 一 种 不 可 独立 应 
用 的 特性 〈 就 象 一 个 switch 语 句 ) ， 只 可 与 其 他 元 系 协 同 使 用 。 我 们 应 
将 其 作为 类 总 体 关 系 的 一 部 分 来 看 竺 。 人 们 经 冲 混 消 Java 其 他 的 、 非 面 
问 对 象 的 特性 ， 比 如 方法 过 载 等 ， 这 些 特性 有 时 也 具有 面 问 对 象 的 菏 些 
RE (EAN BUR TE: 如 果 以 后 没有 绑 定 ， 就 不 成 其 为 多 形 性 。 


为 使 用 多 形 性 万 至 面 回 对 象 的 技术 ， 特 别 是 在 自己 的 程序 中 ， 必 须 将 目 
己 的 编程 视野 扩展 到 不 仅 包括 单独 一 个 类 的 成 员 和 消息 ， 也 要 包括 类 与 
类 之 间 的 一 致 性 以 及 它们 的 关系 。 尽 管 这 要 求学 习 时 付出 更 多 的 精力 ， 
但 却 是 非常 值得 的 ， 因 为 只 有 这 样 才 可 真正 有 效 地 加 快 目 己 的 编程 速 
度 、 更 好 地 组 织 代码 、 更 容易 做 出 包容 面 广 的 程序 以 及 更 易 对 自己 的 代 
码 进行 维护 与 扩展 。 























7.10 练习 


(1) @i/# Rodent (MALIA) :Mouse (CN) Gerbil (itz 
WO „Hamster (AMEN) 等 的 一 个 继承 分 级 结构 。 在 基础 类 中 ， 提 供 适 
用 于 所 有 Rodent 的 方法 ， 并 在 衍生 类 中 缆 盖 它们 ， 从 而 根据 不 同类 型 的 
Rodent 采 取 不 同 的 行动 。 创 建 一 个 Rodent 数 组 ， 在 其 中 填充 不 同类 型 的 
Rodent， 然 后 调用 自己 的 基础 类 方法 ， 看 看 会 有 什么 情况 发 生 。 


(2) 修改 练习 1， 使 Rodent 成 为 一 个 接口 。 
(3) 改正 WindError.java 中 的 问题 。 


(4) 在 GreenhouseControls.java 中 ， 添 加 Event 内 部 类 ， 使 其 能 打开 和 关闭 














BEE 对 象 的 容纳 


“如 末 一 个 程序 从 含有 数量 辕 定 的 对 象 ， 而 且 已 知 它们 的 存在 时 间 ， 那 
么 这 个 程序 可 以 说 是 相当 简单 的 。 


通常 ， 我 们 的 程序 需 ?要 根据 程序 运行 时 才 知 道 的 一 些 标准 创建 新 对 象 。 

在 非 程序 正式 运行 ， 否 则 我 们 根本 不 知道 自己 到 底 需 要 多 少数 量 的 对 

象 ， 甚 至 不 知道 它们 的 准确 类 型 。 为 了 满足 常规 编程 的 需要 ， 我 们 要 求 
能 在 任何 时 候 、 任 何 地 点 创建 任意 数量 的 对 象 。 所 以 不 可 依赖 一 个 已 命 
名 的 句柄 来 容纳 自己 的 每 一 个 对 象 ， 就 象 下 面 这 样 : 


MyObject myHandle; 
因为 根本 不 知道 目 己 实际 需要 多 少 这 样 的 东西 。 


为 解雇 这 个 非常 关键 的 问题 ，Java 提 供 了 容纳 对 象 〈 或 者 对 象 的 句柄 ) 
的 多 种 方式 。 其 中 内 建 的 类 型 是 数组 ， 我 们 之 前 已 讨论 过 它 ， 本 章 准备 
De 此 外 ，Java 的 工具 (实用 程序 ) 库 提供 了 一 

些 “ 和 集合 尔 作 “容器 类 ”， 但 该 术语 已 由 AWT 使 用 ， 所 以 这 里 仍 采 
H a 称呼 〉。 利 用 这 些 集 合 类 ， 我 们 可 以 容纳 乃至 操纵 自己 的 
对 象 。 本 章 的 剩余 部 分 会 就 此 进行 详细 讨论 。 



































8.1 数组 


对 数组 的 大 多 数 必要 的 介绍 已 在 第 4 章 的 最 后 一 节 进 行 。 通 过 那里 的 学 
习 ， 大 家 已 知道 自己 该 如 何 定义 及 初始 化 一 个 数组 。 对 象 的 容纳 是 本 章 
的 重点 ， 而 数组 只 是 容纳 对 象 的 一 种 方式 。 但 由 于 还 有 其 他 大 量 方法 可 
容纳 数组 ， 所 以 是 哪些 地 方 使 数组 显得 如 此 特别 呢 ? 


有 两 方面 的 问题 将 数组 与 其 他 集合 类 型 区 分 开 来 : 效率 和 类 型 。 对 于 
Java 来 说 ， 为 保存 和 访问 一 系列 对 象 〈 实 际 是 对 象 的 句柄 〉 数 组， 最 有 
效 的 方法 莫 过 于 数组 。 数 组 实际 代表 一 个 简单 的 线性 序列 ， 它 使 得 元 素 
的 访问 速度 非常 快 ， 但 我 们 却 要 为 这 种 速度 付出 代价 : 创建 一 个 数组 对 
象 时 ， 它 的 大 小 是 固定 的 ， 而 且 不 可 在 那个 数组 对 象 的 “存在 时 间 ” 内 发 
生 改 变 。 可 创建 特定 大 小 的 一 个 数组 ， 然 后 假如 用 光 了 存储 空间 ， 就 再 
创建 一 个 新 数组 ， 将 所 有 句柄 从 旧 数 组 移 到 新 数组 。 这 属于 “天 

Œ” (Vector) 类 的 行为 ， 本 章 稍 后 还 会 详细 讨论 它 。 然 而 ， 由 于 为 这 种 
大 小 的 灵活 性 要 付出 较 大 的 代价 ， 所 以 我 们 认为 矢量 的 效率 并 没有 数组 


i 
[==] 
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C++ 的 矢量 类 知道 自己 容纳 的 是 什么 类 型 的 对 象 ， 但 同 Java 的 数组 相 
比 ， 它 却 有 一 个 明显 的 缺点 : C++ 矢量 类 的 operator[] 不 能 进行 范围 检 
查 ， 所 以 很 容易 超出 边界 〈 然 而 ， 它 可 以 查询 vector 有 多 大 ， 而 且 at0 方 
法 确实 能 进行 范围 检查 ) 。 在 Java 中 ， 无 论 使 用 的 是 数组 还 是 集合 ， 都 
会 进行 范围 检查 铬 超过 边界 ， 就 会 获得 一 个 RuntimeException( 运 
行 期 违例 ) 错误 。 正 如 大 家 在 第 9 章 会 学 到 的 那样 ， 这 类 违例 指出 的 是 
一 个 程序 员 错 误 ， 所 以 不 需要 在 代码 中 检查 它 。 在 另 一 方面 ， 由 于 
C++ 的 vector 不 进行 范围 检查 ， 所 以 访问 速度 较 快 在 Java 中 ， 由 于 对 
数组 和 集合 都 要 进行 范围 检查 ， 所 以 对 性 能 有 一 定 的 影响 。 


本 间 还 要 学 习 男 外 几 种 常见 的 集合 类 : Vector (RÆ) . Stack CHER) 
VK Hashtable (FJK) 。 这 些 类 都 涉及 对 对 象 的 处 理 好 象 它们 没 
有 特定 的 类 型 。 换 言 之 ， 它 们 将 其 当 作 Object 类 型 处 理 (Object 类 型 是 
Java 中 所 有 类 的 “ 根 ” 类 ) 。 从 某 个 角度 看 ， 这 种 处 理 方法 是 非常 合理 

的 : 我 们 仅 需 构建 一 个 集合 ， 然 后 任何 Java 对 象 都 可 以 进入 那个 集合 

( 除 基 本 数据 类 型 外 可 用 Java 的 基本 类 型 封装 类 将 其 作为 常数 置 入 
集合 ， 或 者 将 其 封装 到 自己 的 类 内 ， 作 为 可 以 变化 的 值 使 用 ) 。 这 再 一 
次 反映 了 数组 优 于 常规 集合 : 创建 一 个 数组 时 ， 可 令 其 容纳 一 种 特定 的 





















































类 型 。 这 意味 着 可 进行 编译 期 类 型 检查 ， 预 防 自己 设置 了 错误 的 类 型 ， 
或 者 错误 指定 了 准备 提取 的 类 型 。 当 然 ， 在 编译 期 或 者 运行 期 ，Java 会 
防止 我 们 将 不 当 的 消息 发 给 一 个 对 象 。 所 以 我 们 不 必 考 虑 自己 的 哪 种 做 
法 更 加 危险， 只 要 编译 器 能 及 时 地 指出 错误 ， 同 时 在 运行 期 间 加 快速 
ee 


考虑 到 执行 效率 和 类型 检查 ， 应 尽 可 能 地 采用 数组 。 然 而 ， 当 我 们 试图 
解决 一 个 更 常规 的 问题 时 ， 数 组 的 局 限 也 可 能 显得 非常 明显 。 在 研究 过 
数组 以 后 ， 本 章 剩余 的 部 分 将 把 重点 放 到 Java 提 供 的 集合 类 身上。 


8.1.1 数组 和 第 一 类 对 象 


无 论 使 用 的 数组 属于 什么 类 型 ， 数 组 标识 符 实际 都 是 指向 真实 对 象 的 一 
个 句柄 。 那 些 对 象 本 里 是 在 内 存 “ 堆 ”里 创建 的 。 堆 对 象 既 可 “ 隐 式 ”创建 
〈 即 默认 产生 ) ， 亦 可 “ 显 式 ”创建 〈 即 明确 指定 ， 用 一 个 new 表 达 
式 ) 。 堆 对 象 的 一 部 分 (实际 是 我 们 能 访问 的 唯一 字段 或 方法 ) 是 只 读 
的 length KE 成员， 它 告诉 我 们 那个 数组 对 象 里 最 多 能 容纳 多 少 元 
素 。 对 于 数组 对 象 ，“[]* 语 法 是 我 们 能 采用 的 唯一 男 类 访问 方法 。 


下 面 这 个 例子 展示 了 对 数组 进行 初始 化 的 不 同方 式 ， 以 及 如 何 将 数组 句 
柄 分 配给 不 同 的 数组 对 象 。 它 也 揭示 出 对 象 数 组 和 基本 数据 类 型 数组 在 
使 用 方法 上 几乎 是 完全 一 致 的 。 唯 一 的 兰 别 在 于 对 象 数组 容纳 的 是 名 
顶 ， 而 基本 数据 类 型 数组 容纳 的 是 具体 的 数值 (大 在 执行 此 程序 时 过 到 
困难 ， 请 参考 第 3 间 的 “赋值 ”小 市 ): 


//: ArraySize.java 




















// Initialization & re-assignment of arrays 
package c08; 
Class Weeble {} // A small mythical creature 
public class ArraySize { 

public static void main(String[] args) { 


// Arrays of objects: 


Weeble[] a; // Null handle 
Weeble[] b = new Weeble[5]; // Null handles 
Weeble[] c = new Weeble[4]; 
for(int i = 0; i < c.length; i++) 
c[i] = new Weeble(); 
Weeble[] d = { 
new Weeble(), new Weeble(), new Weeble( ) 
}; 
// Compile error: variable a not initialized: 
//\System.out.printin("a.length=" + a.length); 
System.out.printin("b.length = " + b.length); 
// The handles inside the array are 
// automatically initialized to null: 
for(int 1 = 0; i < b.length; i++) 


System.out.printin("b[" + i + "]=" + b[i]); 


System.out.printin("c.length = " + c.length); 
System.out.println("d.length = " + d.length); 
a =d; 

System.out.println("a.length = " + a.length); 


// Java 1.1 initialization syntax: 
a = new Weeble[] { 


new Weeble(), new Weeble() 


}; 


System.out.printin("a.length = " + a.length); 
// Arrays of primitives: 
int[] e; // Null handle 
int[] f = new int[5]; 
int[] g = new int[4]; 
for(int i = 0; i < g.length; i++) 
g[i] = i*i; 
int[] h = { 11, 47, 93 }; 
// Compile error: variable e not initialized: 
//\System.out.printin("e.length=" + e.length); 
System.out.printin("f.length = " + f.length); 
// The primitives inside the array are 
// automatically initialized to zero: 
for(int i = 0; i < f.length; i++) 


System.out.printin("f[" + i + "j=" + f[i]); 


System.out.println("g.length = " + g.length); 
System.out.println("h.length = " + h.length); 
e = h; 

System.out.println("e.length = " + e.length); 


// Java 1.1 initialization syntax: 
e = new int[] { 1, 2 }; 


System.out.printin("e.length = " + e.length); 


} ///:~ 


Here’s the output from the program: 


b.length = 5 


b[0]=null 
b[1]J=null 
b[2]=null 
b[3]=null 
b[4]=null 
c.length = 4 
d.length = 3 
a.length = 3 
a.length = 2 
f.length = 5 
F[0]=0 
f[1]=0 
f[2]=0 
F[3]=0 
f[4]=0 
g.length = 4 
h.length = 3 
e.length = 3 


e.length = 2 


其 中 ， 数 组 a 只 是 初始 化 成 一 个 nu 句柄。 此 时 ， 编 译 器 会 禁止 我 们 对 这 
个 句柄 作 任 何 实际 操 作 ， 除 非 已 正确 地 初始 化 了 它 。 数 组 b 被 初始 化 成 
指 加 由 Weeble 句 柄 构成 的 一 个 数组 ， 但 那个 数组 里 实际 并 未 放置 任何 

Weeble 对 象 。 然 而 ， 我 们 仍然 可 以 查询 那个 数组 的 大 小 ， 因 为 b 指 癌 的 
是 一 个 合法 对 象 。 这 也 为 我 们 带 来 了 一 个 难题 : 不 可 知道 那个 数组 里 实 
际 包 含 了 多 少 个 元 素 ， 因 为 length 只 告诉 我 们 可 将 多 少 元 素 置 入 那个 数 
组 。 换 言 之 ， 我 们 只 知道 数组 对 象 的 大 小 或 容量 ， 不 知 其 实际 容纳 了 多 
少 个 元 素 。 尺 管 如 此 ， 由 于 数组 对 象 在 创建 之 初 会 自动 初始 化 成 null， 

所 以 可 检查 它 是 否 为 nul， 判 断 一 个 特定 的 数组 “空位 ”是 否 容纳 一 个 对 
象 。 类 似 地 ， 由 基本 数据 类 型 构成 的 数组 会 自动 初始 化 成 零 〈 针 对 数值 
类 型 ) noul 〈 字 符 类 型 ) 或 者 false《〈 布 尔 类 型 ) 。 


数组 < 显示 出 我 们 首先 创建 一 个 数组 对 象 ， 再 将 Weeble 对 象 赋 给 那个 数 
组 的 所 有 “空位 ”。 数 组 d 揭 示 出 “集合 初始 化 ”语法 ， 从 而 创建 数组 对 象 
(用 new 命 令 明 确 进行 ， 类 似 于 数组 c) ， 然 后 用 Weeble 对 象 进 行 初 始 
化 ， 全 部 工作 在 一 条 语句 里 完成 。 


下 面 这 个 表达 式 : 

a=d; 

同 我 们 展示 了 如 何 取得 同一 个 数组 对 象 连接 的 句 顶 ， 然 后 将 其 赋 给 男 一 
个 数组 对 象 ， 融 象 我 们 针对 对 象 句柄 的 其 他 任何 类 型 做 的 那样 。 现 在 ， 

a 和 d 都 指向 内 存 堆 内 同样 的 数组 对 象 。 

Java 1.1 加 入 了 一 种 新 的 数组 初始 化 语法 ， 可 将 其 想象 成 “动态 集合 初始 
化 ”。 由 d 采 用 的 Java 1.0 集 合 人 初始 化 方法 则 必须 在 定义 d 的 同时 进行 。 但 
若 采用 Java 。 “1.1 的 语法 ， 却 可 以 在 任何 地 方 创 建 和 初始 化 一 个 数组 对 
象 。 例 如 ， 假 设 hide0) 方 法 用 于 取得 一 个 Weeble 对 象 数 组 ， 那 么 调用 它 

时 传统 的 方法 是 : 

hide(d); 

但 在 Java 1.1 中 ， 亦 可 动态 创建 想 作为 参数 传递 的 数组 ， 如 下 所 示 : 


hide(new Weeble[] {new Weeble(), new Weeble() }); 














这 一 新 式 语法 使 我 们 在 某 些 场合 下 写 代 码 更 方便 了 。 


上 述 例子 的 第 二 部 分 揭示 出 这 样 一 个 问题 : 对 于 由 基本 数据 类 型 构成 的 
数组 ， 它 们 的 运作 方式 与 对 象 数 组 极为 相似 ， 只 是 前 者 直接 包容 了 基本 
类 型 的 数据 值 。 


1. 基本 数据 类 型 集合 


集合 类 只 能 容纳 对 象 句柄 。 但 对 一 个 数组 ， 却 既 可 令 其 直接 容纳 基本 类 
型 的 数据 ， 亦 可 容纳 指向 对 象 的 句柄 。 利 用 象 Imteger、Double 之 类 

的 “封装 器 ”类 ， 可 将 基本 数据 类 型 的 值 置 入 一 个 集合 里 。 但 正如 本 章 后 
面 会 在 WordCount.java 例 子 中 讲 到 的 那样 ， 用 于 基本 数据 类 型 的 封装 器 
类 只 是 在 某 些 场合 下 才能 发 挥 作 用 。 无 论 将 基本 类 型 的 数据 置 入 数组 ， 
还 是 将 其 封装 进入 位 于 集合 的 一 个 类 内 ， 都 涉及 到 执行 效率 的 问题 。 显 
然 ， 知 能 创建 和 访问 一 个 基本 数据 类 型 数组 ， 那 么 比 起 访问 一 个 封装 数 
据 的 集合 ， 前 者 的 效率 会 高 出 许多 。 


当然 ， 假 如 准备 一 种 基本 数据 类 型 ， 同 时 又 想 要 集合 的 灵活 性 (在 需要 
的 时 候 可 自动 扩展 ， 腾 出 更 多 的 空间 〉 ， 就 不 宜 使 用 数组 ， 必 须 使 用 由 
封装 的 数据 构成 的 一 个 集合 。 大 家 或 许 认 为 针对 每 种 基本 数据 类 型 ， 都 
应 有 一 种 特殊 类 型 的 Vector。 但 Java 并 未 提供 这 一 特性 。 某 些 形式 的 建 
模 机 制 或 许 会 在 某 一 天 帮助 Java 更 好 地 解决 这 个 问题 〈 注 释 GD) 。 


QD: 这 儿 是 C++ 比 Java 做 得 好 的 一 个 地 方 ， 因 为 C++ 通过 template 关 键 字 
提供 了 对 “参数 化 类 型 > 的 文 持 。 


8.1.2 数组 的 返回 


假定 我 们 现在 想 写 一 个 方法 ， 同 时 不 希望 它 仅 仅 返 回 一 样 东 西 ， 而 是 想 
返回 一 系列 东西 。 此 时 ， 象 C 和 C++ 这 样 的 语言 会 使 问题 复杂 化 ， 因 为 
我 们 不 能 返回 一 个 数组 ， 只 能 返回 指 加 数组 的 一 个 指针 。 这 样 就 非常 麻 
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现 。 


Java 采 用 的 是 类 似 的 方法 ， 但 我 们 能 “返回 一 个 数组 ”。 当 然 ， 此 时 返回 
的 实际 仍 是 指向 数组 的 指针 。 但 在 Java 里 ， 我 们 永远 不 必 担 心 那个 数组 
的 是 否 可 用 一 一 只 要 需要 ， 它 就 会 自动 存在 。 而 且 垃 圾 收集 占 会 在 我 们 
完成 后 目 动 将 其 清除 。 




















作为 一 个 例子 ， 请 思考 如 何 返 回 一 个 字 串 数组 : 


//: IceCcream,java 


// Returning arrays from methods 
public class IceCream { 
static String[] flav = { 
"Chocolate", "Strawberry", 
"Vanilla Fudge Swirl", "Mint Chip", 
"Mocha Almond Fudge", "Rum Raisin", 
"Praline Cream", "Mud Pie" 
}; 
static String[] flavorSet(int n) { 
// Force it to be positive & within bounds: 
n = Math.abs(n) % (flav.length + 1); 
String[] results = new String[n]; 
int[] picks = new int[n]; 
for(int 1 = 0; 1 < picks.length; i++) 
picks[i] = -1; 
for(int i = 0; i < picks.length; i++) { 
retry: 
while(true) { 
int t = 


(int)(Math.random() * flav.length); 


for(int j = 0; j < i; j++) 
if(picks[j] == t) continue retry; 

picks[1i] = t; 

results[i] = flav[t]; 


break; 


} 
return results; 
} 
public static void main(String[] args) { 
for(int i = 0; i < 20; itt) { 
System.out.printin( 
"flavorSet(" + i + ") = "); 
String[] fl = flavorSet(flav.length); 
for(int j = 0; j < fl.length; j++) 


System.out.println("\t" + f1[j]); 


} MAS i~ 


flavorSet() 方 法 创建 了 一 个 名 为 results 的 String 数 组 。 该 数组 的 大 小 为 n 
有 具体 数值 取 诀 于 我 们 传递 给 方法 的 自 变量 。 随 后 ， 它 从 数组 flav 里 
随机 挑选 一 些 “ 香 料 ”(Flavor) ， 并 将 它们 置 入 results 里 ， 并 最 终 返 回 
results。 返 回 数组 与 返回 其 他 任何 对 象 没 什么 区 别 最 终 返 回 的 都 是 
一 个 句柄 。 至 于 数组 到 底 是 在 flavorSet() 里 创建 的 ， 还 是 在 其 他 什么 地 








方 创建 的 ， 这 个 问题 并 不 重要 ， 因 为 反正 返回 的 仅 是 一 个 句柄 。 一 旦 我 
们 的 操作 完成 ， 垃 圾 收集 器 会 目 动 关照 数组 的 清除 工作 。 而 且 只 要 我 们 
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另 一 方面 ， 注 意 当 flavorSetO 随 机 挑选 香料 的 时 候 ， 它 需要 保证 以 前 出 
现 过 的 一 次 随机 选择 不 会 再 次 出 现 。 为 达到 这 个 目的 ， 它 使 用 了 一 个 无 
限 while 循 环 ， 不 断 地 作出 随机 选择 ， 直 到 发 现 未 在 picks 数 组 里 出 现 过 
的 一 个 元 素 为 止 (当然 ， 也 可 以 进行 字 串 比较 ， 检 查 随机 选择 是 否 在 
results 数 组 里 出 现 过 ， 但 字 串 比较 的 效率 比较 低 ) 。 闪 成功， 束 添 加 这 
个 元 素 ， 并 中 断 循环 〈break) ， 再 查找 下 一 个 《iji 值 会 递增 ) . (AA 
是 一 个 已 在 picks 里 出 现 过 的 数组 ， 就 用 标签 式 的 continue 往 回 跳 两 级 ， 
强制 选择 一 个 新 t。 用 一 个 调试 程序 可 以 很 清楚 地 看 到 这 个 过 程 。 


main0 能 显示 出 20 个 完整 的 香料 集合 ， 所 以 我 们 看 到 favorSetO 每 次 都 用 
一 个 随机 顺序 选择 香料 。 为 体会 这 一 点 ， 最 简单 的 方法 就 是 将 输出 重 导 
向 进入 一 个 文件 ， 然 后 直接 观看 这 个 文件 的 内 容 。 


8.2 集合 


现在 总 结 一 下 我 们 前 面 学 过 的 东西 : 为 容纳 一 组 对 象 ， 最 适宜 的 选择 应 
当 是 数组 。 而 且 假 如 容纳 的 是 一 系列 基本 数据 类 型 ， 更 是 必须 采用 数 
组 。 在 本 章 剩 下 的 部 分 ， 大 家 将 接触 到 一 些 更 稼 规 的 情况 。 当 我 们 编写 
程序 时 ， 通 党 并 不 能 确切 地 知道 最 终 需 要 多 少 个 对 象 。 有 些 时 候 甚 至 想 
用 更 复杂 的 方式 来 保存 对 象 。 为 解决 这 个 问题 ，Java 提 供 了 四 种 类 型 
的 “集合 类 ”: Vector (RÆ) 、BitSet MÆ) 、Stack (HERE) 以 及 
Hashtable〈 散 列表 ) 。 与 拥有 集合 功能 的 其 他 语言 相 比 ， 尽 管 这 儿 的 数 
量 显 得 相当 少 ， 但 仍然 能 用 它们 解决 数量 惊人 的 实际 问题 。 


这 些 和 集合 类 具有 形形色色 的 特征 。 例 如 ，Stack 实 现 了 一 个 LIFO 先入 
先 出 〉 序 列 ， 而 Hashtable 是 一 种 “关联 数 组 *”， 人 允许 我 们 将 任何 对 象 关联 
起 来 。 除 此 以 外 ， 所 有 Java 集 合 类 都 能 目 动 改变 目 身 的 大 小 。 所 以 ， 我 
| 合 弄 得 有 多 

















8.2.1 缺点 : 类 型 未 知 


使 用 Java 集 合 的 “缺点 ?是 在 将 对 象 置 入 一 个 集合 时 丢失 了 类 型 信息 。 之 
所 以 会 发 生 这 种 情况 ， 是 由 于 当初 编写 集合 时 ， 那 个 集合 的 程序 员 根 本 
不 知道 用 户 到 底 想 把 什么 类 型 置 入 集合 。 厂 指示 杂 个 集合 只 允许 特定 的 
类 型 ， 会 妨碍 它 成 为 一 个 “常规 用 途 ” 的 工具 ， 为 用 户 剖 来 麻烦 。 为 解决 
这 个 问题 ， 集 合 实 际 容纳 的 是 类 型 为 Object 的 一 些 对 象 的 句柄 。 这 种 类 
型 当然 代表 Java 中 的 所 有 对 象 ， 因 为 它 是 所 有 类 的 根 。 当 然 ， 也 要 注意 
这 并 不 包括 基本 数据 类 型 ， 因 为 它们 并 不 是 从 “任何 东西 ”继承 来 的 。 这 
征 一 个 很 好 的 方案 ， 只 是 不 适用 下 述 场合 : 


(1) 将 一 个 对 象 句柄 置 入 集合 时 ， 由 于 类 型 信息 会 被 抛弃 ， 所 以 任何 类 
型 的 对 象 都 可 进入 我 们 的 集合 一 一 即便 特别 指示 它 只 能 容纳 特定 类 型 的 
对 象 。 举 个 例子 来 说 ， 虽 然 指示 和 它 只 能 容纳 猫 ， 但 事实 上 任何 人 都 可 以 
把 一 条 狗 扔 进来 。 


(2) ”由 于 类 型 信息 不 复 存在 ， 所 以 集合 能 肯定 的 唯一 事情 就 是 自己 容纳 
的 是 指向 一 个 对 象 的 句柄 。 正 式 使 用 它 之 前 ， 必 须 对 其 进行 造型 ， 使 其 
具有 正确 的 类 型 。 




















值得 欣慰 的 是 ，Java 不 允许 人 们 滥用 置 入 集合 的 对 象 。 假 如 将 一 条 狗 扔 
进 一 个 猫 的 集合 ， 那 么 仍 会 将 集合 内 的 所 有 东西 都 看 作 猎 ， 所 以 在 使 用 
那 条 狗 时 会 得 到 一 个 “违例 ”错误 。 在 同样 的 意义 上 ， 假 耕 试 图 将 一 条 狗 
的 句柄 “造型 ?到 一 只 猫 ， 那 么 运行 期 间 仍 会 得 到 一 个 “违例 ”错误 。 

下 面 是 个 例子 : 


//: CatsAndDogs.java 


// Simple collection example (Vector) 
import java.util.*; 
class Cat { 

private int catNumber,; 

Cat(int i) { 

catNumber = i; 
} 
void print() { 


System.out.printin("Cat #" + catNumber); 


} 
class Dog { 
private int dogNumber ; 
Dog(int i) { 
dogNumber = i; 
} 


void print() { 


System.out.printin("Dog #" + dogNumber ) ; 
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public class CatsAndDogs { 
public static void main(String[] args) { 
Vector cats = new Vector(); 
for(int i = 0; i < 7; i++) 
cats.addElement(new Cat(i)); 
// Not a problem to add a dog to cats: 
cats.addElement(new Dog(7)); 
for(int i = 0; i < cats.size(); i++) 
((Cat)cats.elementAt(i)).print(); 
// Dog is detected only at run-time 
} 
} ///:~ 


可 以 看 出 ，Vector 的 使 用 是 非常 简单 的 ， 先 创建 一 个 ， 再 用 addElement() 
置 入 对 象 ， 以 后 用 elementAt() 取 得 那些 对 象 ( 注 意 Vector 有 一 个 size() 方 
可 使 我 们 知道 已 添加 了 多 少 个 元 素 ， 以 便 防 止 误 超 边界 ， 造 成 违例 
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Cat 和 Dog 类 都 非常 浅显 除了 都 是 “对 象 * 之 外 ， 它 们 并 无 特别 之 处 
《 倘 知 不 明确 指出 从 什么 类 继承 ， 就 默认 为 从 Object 继承 。 所 以 我 们 不 
仅 能 用 Vector 方法 将 Cat 对 象 置 入 这 个 集合 ， 也 能 添加 Dog 对 象 ， 同 时 不 
会 在 编译 期 和 运行 期 得 到 任何 出 错 提示 。 用 Vector 方 法 elementAt() 获 取 
原本 认为 是 Cat 的 对 象 时 ， 实 际 获得 的 是 指 同 一 个 Object 的 句柄 ， 必 须 将 
那个 对 象 造 型 为 Cat。 随 后 ， 需 要 将 整个 表达 式 用 括号 封闭 起 来 ， 在 为 














Cat 调 用 print0 方 法 之 前 进行 强制 造型 ， 否 则 就 会 出 现 一 个 语法 错误 。 在 
运行 期 间 ， 如 果 试图 将 Dog 对 象 造型 为 Cat， 就 会 得 到 一 个 违例 。 


这 些 处 理 的 意义 都 非常 深远 。 尺 管 显得 有 些 麻烦 ， 但 却 获得 了 安全 上 的 
保证 。 我 们 从 此 再 难 侦 然 造成 一 些 隐藏 得 深 的 错误 。 寿 程序 的 一 个 部 分 
(或 几 个 部 分 ) 将 对 象 插 入 一 个 集合 ， 但 我 们 只 是 通过 一 次 违例 在 程序 
的 某 个 部 分 发 现 一 个 错误 的 对 象 置 入 了 集合 ， 就 必须 找 出 插入 错误 的 位 
置 。 当 然 ， 可 通过 检查 代码 达到 这 个 目的 ， 但 这 或 许 是 最 笨 的 调试 工 
具 。 男 一 方面 ， 我 们 可 从 一 些 标准 化 的 集合 类 开始 自己 的 编程 。 尺 管 它 
们 在 功能 上 存在 一 些 不 足 ， 且 显得 有 些 笨 拙 ， 但 却 能 保证 没有 隐藏 的 错 
误 。 


1. 错误 有 时 并 不 显露 出 来 


在 东 些 情况 下 ， 程 序 似乎 正确 地 工作 ， 不 造型 回 我 们 原来 的 类 型 。 第 一 
种 情况 是 相当 特殊 的 ，String 类 从 编译 占 获 得 了 额外 的 帮助 ， 使 其 能 够 
正常 工作 。 只 有 要 编译 器 期 待 的 是 一 个 String 对 象 ， 但 它 没 有 得 到 一 个 ， 

就 会 自动 调用 在 Object 里 定义 、 并 且 能 够 由 任何 Java 类 履 兰 的 toString() 
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因此 ， 为 了 让 上 自己 类 的 对 象 能 显示 出 来 ， 要 做 的 全 部 事情 就 是 窗 盖 
toString) IE, BR BIRT: 


//: WorksAnyway.java 


























// In special cases, things just seem 
// to work correctly. 
import java.util.*; 
class Mouse { 
private int mouseNumber ; 
Mouse(int i) { 


mouseNumber = i; 


} 
// Magic method: 
public String toString() { 

return "This is Mouse #" + mouseNumber ; 

} 

void print(String msg) { 
if(msg != null) System.out.printiln(msg); 
System.out.printin( 


"Mouse number " + mouseNumber ); 
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class MouseTrap { 
static void caughtYa(Object m) { 
Mouse mouse = (Mouse)m; // Cast from Object 


mouse.print("Caught one!"); 


} 
public class WorksAnyway { 
public static void main(String[] args) { 
Vector mice = new Vector(); 
for(int i = 0; i < 3; i++) 
mice.addElement(new Mouse(i)); 


for(int i = 0; i < mice.size(); i++) { 


// No cast necessary, automatic call 
// to Object.toString(): 
System.out.printin( 
"Free mouse: " + mice.elementAt(i)); 


MouseTrap.caughtYa(mice.elementAt(i)); 


} 
SA 


可 在 Mouse 里 看 到 对 toString() 的 重 定 义 代 码 。 在 main() 的 第 二 个 for 循 环 
中 ， 可 发 现下 述 语句 : 


System.out.println("Free mouse: " + 
mice.elementAt(i)); 


在 “+” 后 ， 编 译 占 预期 看 到 的 是 一 个 String 对 象 。elementAt() 生 成 了 一 个 
Object， 所 以 为 获得 希望 的 String， 编 译 器 会 默认 调用 toString()。 但 不 骏 
的 是 ， 只 有 针对 String 才 能 得 到 象 这 样 的 结果 ， 其 他 任何 类 型 都 不 会 进 
行 这 样 的 转换 。 


隐藏 造型 的 第 二 种 方法 已 在 Mousetrap 里 得 到 了 应 用 。caughtYa() 方 法 接 
收 的 不 是 一 个 Mouse， 而 是 一 个 Object。 随 后 再 将 其 造型 为 一 个 Mouse。 
当然 ， 这 样 做 是 非常 冒失 的 ， 因 为 通过 接收 一 个 Object， 任 何 东西 都 可 
以 传递 给 方法 。 然 而 ， 假 知 造 型 不 正确 一 如 果 我 们 传递 了 错误 的 类 型 
就 会 在 运行 期 间 得 到 一 个 违例 错误 。 这 当然 没有 在 编译 期 进行 检查 
好 ， 但 仍然 能 防止 问题 的 发 生 。 注 意 在 使 用 这 个 方法 时 考 需 进行 造型 : 











MouseTrap.caughtYa(mice.elementAt(i)); 
2. 生成 能 自动 判别 类 型 的 Vector 
大 家 或 许 不 想 放弃 刚才 那个 问题 。 一 个 更 “健壮 ”的 方案 是 用 Vector 创建 


使 其 只 接收 我 们 指定 的 类 型 ， 也 只 生成 我 们 希望 的 类 型 。 如 
下 所 未: 


//: GopherVector.java 


// A type-conscious Vector 
import java.util.*; 
class Gopher { 
private int gopherNumber; 
Gopher(int i) { 
gopherNumber = i; 
} 
void print(String msg) { 
if(msg != null) System.out.printiln(msg); 
System.out.printin( 


"Gopher number " + gopherNumber ); 


} 


class GopherTrap { 
static void caughtYa(Gopher g) { 


g.print("Caught one!"); 


} 


Class GopherVector { 


private Vector v = new Vector(); 
public void addElement(Gopher m) { 
v.addElement(m); 
} 
public Gopher elementAt(int index) { 
return (Gopher)v.elementAt(index); 
} 
public int size() { return v.size(); } 
public static void main(String[] args) { 
GopherVector gophers = new GopherVector(); 
for(int i = 0; i < 3; i++) 
gophers.addElement(new Gopher(i)); 
for(int i = 0; i < gophers.size(); i++) 
GopherTrap.caughtYa(gophers.elementAt(i)); 
} 
} ///:~ 


这 前 一 个 例子 类 似 ， 只 是 新 的 GopherVector 类 有 一 个 类 型 为 Vector 的 
private 成 员 〈 从 Vector 继 承 有 些 厂 烦 ， 理 由 稍 后 便 知 ) ， 而 且 方法 也 和 
Vector 类 似 。 然 而 ， 它 不 会 接收 和 产生 普通 Object， 只 对 Gopher 对 象 感 
兴趣 。 


由 于 GopherVector 只 接收 一 个 Gopher Gh O ， 上 所 以 假如 我 们 使 用 : 
gophers.addElement(new Pigeon()); 
就 会 在 编译 期 间 获 得 一 条 出 错 消息 。 采 用 这 种 方式 ， 尽 管 从 编码 的 角度 








看 显得 更 令 人 沉 问 ， 但 可 以 立即 判断 出 是 否 使 用 了 正确 的 类 型 。 
注意 在 使 用 elementAt() 时 不 必 进 行 造型 一 一 它 肯 定 是 一 个 Gopher。 
3. 参数 化 类 型 


这 类 问题 并 不 是 孤立 的 一 一 我 们 许多 时 候 都 要 在 其 他 类 型 的 基础 上 创建 
新 类 型 。 此 时 ， 在 编译 期 间 拥有 特定 的 类 型 信息 是 非常 有 帮助 的 。 这 便 
是 “参数 化 类 型 * 的 概念 。 在 C++ 中 ， 它 由 语言 通过 “模板 ”获得 了 直接 文 
持 。 至 少 ，Java 保 留 了 关键 字 generic， 期 望 有 一 天 能 够 支持 参数 化 类 
型 。 但 我 们 现在 无 法 确定 这 一 天 何 时 会 来 临 。 














8.3 枚 举 器 《反复 器 ) 


在 任何 集合 类 中 ， 必 须 通 过 某 种 方法 在 其 中 置 入 对 象 ， 再 用 另 一 种 方法 
从 中 取得 对 象 。 毕 竟 ， 容 纳 各 种 各 样 的 对 象 正 是 集合 的 首要 任务 。 在 
Vector 中 ，addElementO 便 是 我 们 插入 对 象 采 用 的 方法 ， 而 elementAtO 是 
提取 对 象 的 唯一 方法 。Vector 非 党 灵活 ， 我 们 可 在 任何 时 候选 择 任何 东 
西 ， 并 可 使 用 不 同 的 索引 选择 多 个 元 素 。 


若 从 更 高 的 角度 看 这 个 问题 ， 就 会 发 现 它 的 一 个 缺陷 : 需要 事先 知道 集 
合 的 准确 类 型 ， 否 则 无 法 使 用 。 乍 看 来 ， 这 一 点 似乎 没什么 关系 。 但 假 
若 最 开始 决定 使 用 Vector， 后 来 在 程序 中 又 决定 〈 考 虑 执行 效率 的 原 
Al) 改变 成 一 个 List〈 属 于 Javal1.2 集 合 库 的 一 部 分 ) ， 这 时 又 该 如 何 做 
呢 ? 


可 利用 “反复 器 ”(〈Iterator) 的 概念 达到 这 个 目的 。 它 可 以 是 一 个 对 象 ， 

作用 是 壳 历 一 系列 对 象 ， 并 选择 那个 序列 中 的 每 个 对 象 ， 同 时 不 让 客户 
程序 员 知 道 或 关注 那个 序列 的 基础 结构 。 此 外 ， 我 们 通常 认为 反复 器 是 
一 种 “ 轻 量 级 ”对象 ， 也 就 是 说 ， 创 建 它 只 需 付 出 极 少 的 代价 。 但 也 正 是 
由 于 这 个 原因 ， 我 们 常 发 现 反 复 器 存在 一 些 似乎 很 奇怪 的 限制 。 例 如 ， 

有 些 反 复 器 只 能 隅 一 个 方 同 移动 。 

Java 的 Enumeration (WA, FRAD) 便 是 具有 这 些 限 制 的 一 个 反复 器 的 
例子 。 除 下 面 这 些 外 ， 不 可 再 用 它 做 其 他 任何 事情 : 


(1) 用 一 个 名 为 elements() 的 方法 要 求 集 合 为 我 们 提供 一 个 Enumeration。 
我 们 首次 调用 它 的 nextElement() 时 ， 这 个 Enumeration 会 返回 序列 中 的 第 
=A e 




















(2) 用 nextElement() 获 得 下 一 个 对 象 。 

(3) 用 hasMoreElements() 检 查 序 列 中 是 否 还 有 更 多 的 对 象 。 

@:“ 反 复 器 * 这 个 词 在 C++ 和 OOP 的 其 他 地 方 是 经 常 出 现 的 ， 所 以 很 难 
确定 为 什么 Java 的 开发 者 采用 了 这 样 一 个 奇怪 的 名 字 。Java 1.2 的 集合 库 
修正 了 这 个 问题 以 及 其 他 许多 问题 。 


只 可 用 Enumeration 做 这 些 事 情 ， 不 能 再 有 更 多 。 它 属于 反复 器 一 种 简单 











的 实现 方式 ， 但 功能 依然 十 分 强大 。 为 体会 它 的 运作 过 程 ， 让 我 们 复习 

一 下 本 章 早 些 时候 提 到 的 CatsAndDogs.java 程 序 。 在 原始 版 本 中 ， 

元素， 但 在 下 述 修订 版 中 ， 可 看 到 使 用 
一 也 c y an 


//: CatsAndDogs2.java 








// Simple collection with Enumeration 
import java.util.*; 
class Cat2 { 

private int catNumber,; 

Cat2(int 1) { 

catNumber = i; 
} 
void print() { 


System.out.printin("Cat number " +catNumber); 


} 
class Dog2 { 
private int dogNumber ; 
Dog2(int i) { 
dogNumber = i; 
} 
void print() { 


System.out.printin("Dog number " +dogNumber ); 


} 
public class CatsAndDogs2 { 
public static void main(String[] args) { 
Vector cats = new Vector(); 
for(int i = 0; i < 7; i++) 
cats.addElement(new Cat2(i)); 
// Not a problem to add a dog to cats: 
cats.addElement(new Dog2(7)); 
Enumeration e = cats.elements(); 
while(e.hasMoreElements() ) 
((Cat2)e.nextElement()).print(); 
// Dog is detected only at run-time 
} 
} ///:~ 





我 们 看 到 唯一 的 改变 就 是 最 后 几 行 。 不 再 是 : 
for(int i = 0; i < cats.size(); i++) 
((Cat)cats.elementAt(i)).print(); 

而 是 用 一 个 Enumeration 人 遍历 整个 序列 : 
while(e.hasMoreElements()) 


((Cat2)e.nextElement()).print(); 


使 用 Enumeration， 我 们 不 必 关 心 集合 中 的 元 素数 量 。 所 有 工作 均 由 
hasMoreElements()#nextElement() H IE E T- 
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//: HamsterMaze.java 


// Using an Enumeration 
import java.util.*; 
class Hamster { 
private int hamsterNumber ; 
Hamster(int i) { 
hamsterNumber = i; 
} 
public String toString() { 


return "This is Hamster #" + hamsterNumber ， 


} 
class Printer { 
static void printAll(Enumeration e) { 
while(e.hasMoreElements()) 
System.out.printin( 


e.nextElement().toString()); 


public class HamsterMaze { 
public static void main(String[] args) { 
Vector v = new Vector(); 
for(int i = 0; i < 3; i++) 
v.addElement(new Hamster(i)); 
Printer.printAll(v.elements()); 


} 
SA 


仔细 研究 一 下 打印 方 汰 : 


static void printAll(Enumeration e) { 


while(e.hasMoreElements() ) 
System.out.println( 


e.nextElement().toString()); 


主意 其 中 没有 与 序列 类 型 有 关 的 信息 。 我 们 拥有 的 全 部 东西 便 是 
a 为 了 解 有 关 序 列 的 情况 ， 一 个 Enumeration 便 足够 了 : 可 
取得 下 一 个 对 象 ， 亦 可 知道 是 否 已 抵达 了 末尾 。 取 得 一 系列 对 象 ， 然 后 
在 其 中 遍历， 从 而 执行 一 个 特定 的 操作 一 一 这 是 一 个 版 有 价值 的 编程 概 
念 ， 本 书 许 多 地 方 都 会 沿用 这 一 思路 。 


这 个 看 似 特殊 的 例子 甚至 可 以 更 为 通用 ， 因 为 它 使 用 了 各 规 的 toString0) 
方法 (之 所 以 称 为 常规 ， 是 由 于 它 属于 Object 类 的 一 部 分 ) 。 下 面 是 调 
用 打印 的 男 一 个 方法 (尽管 在 效率 上 可 能 会 差 一 些 ) : 














System.out.println("" + e.nextElement()); 


它 采 用 了 封装 到 Java 内 部 的 “自动 转换 成 字 串 ”技术 。 一 旦 编译 器 碰 到 一 
个 字 串 ， 后 面 跟随 一 个 “+”， 就 会 希望 后 面 又 跟随 一 个 字 捉 ， 并 自动 调 

用 toString()。 在 Java 1.1 中 ， 第 一 个 字 串 是 不 必要 的 ; 所 有 对 象 都 会 转换 
成 字 串 。 亦 可 对 此 执行 一 次 造型 ， 获 得 与 调用 toString0 同 样 的 效果 ; 





System.out.printIn((String)e.nextElement()) 


但 我 们 想 做 的 事情 通常 并 不 仅仅 是 调用 Object 方法 ， 所 以 会 再 度 面 临 类 
型 造型 的 问题 。 对 于 目 己 感 兴趣 的 类 型 ， 必 须 假 定 目 己 已 获得 了 一 个 
Enumeration， 然 后 将 结果 对 象 造 型 成 为 那 种 类 型 〈 重 操作 错误 ， 会 得 到 
运行 期 违例 ) 。 








8.4 集合 的 类 型 


标准 Java 1.0 和 1.1 库 配套 提供 了 非常 少 的 一 系列 集合 类 。 但 对 于 目 己 的 
大 多 数 编程 有 要求， 它们 基本 上 都 能 胜任 。 正 如 大 家 到 本 章 末 尾 会 看 到 
的 ，Java 1.2 提 供 的 是 一 套 重 新 设计 过 的 大 型 集合 库 。 


8.4.1 Vector 


Vector 的 用 法 很 简单 ， 这 已 在 前 面 的 例子 中 得 到 了 证 明 。 尽 管 我 们 大 多 
数 时 候 只 需 用 addElement() 插 入 对 象 ， 用 elementAt() 一 次 提取 一 个 对 

象 ， 并 用 elements() 获 得 对 序列 的 一 个 “ 枚 举 ”。 但 仍 有 其 他 一 系列 方法 是 
非常 有 用 的 。 同 我 们 对 于 Java 库 惯 第 的 做 法 一 样 ， 在 这 里 并 不 使 用 或 讲 
述 所 有 这 些 方法 。 但 请 务必 阅读 相应 的 电子 文档 ， 对 它们 的 工作 有 一 个 
大 概 的 认识 。 

1. 月 尝 Java 

Java 标 准 集合 里 包含 了 toString0) 方 法 ， 所 以 它们 能 生成 自己 的 String 表 达 
方式 ， 包 括 它 们 容纳 的 对 象 。 例 如 在 Vector 中 ，toString0 会 在 Vector 的 

各 个 元 素 中 步 进 和 遍历 ， 并 为 每 个 元 素 调 用 toString0。 假 定 我 们 现在 想 
打印 出 自己 类 的 地 址 。 看 起 来 似乎 简单 地 引用 this 即 可 《特别 是 C++ 程 

序 员 有 这 样 做 的 倾 问 ) : 


//: CrashJava.java 


// One way to crash Java 
import java.util.*; 
public class CrashJava { 
public String toString() { 
return "CrashJava address: " + this + "\n"; 


} 


public static void main(String[] args) { 


Vector v = new Vector( ) ; 

for(int i = 0; i < 10; i++) 
v.addElement(new CrashJava()); 

System.out.println(v); 


I 
} ///:~ 


若 只 是 简单 地 创建 一 个 CrashJava 对 象 ， 并 将 其 打印 出 来 ， 就 会 得 到 无 穷 
无 尽 的 一 系列 违例 错误 。 然 而 ， 假 如 将 CrashJava 对 象 置 入 一 个 Vector， 
并 象 这 里 演示 的 那样 打印 Vector， 束 不 会 出 现 什么 错误 提示 ， 甚 至 连 一 
个 违例 都 不 会 出 现 。 此 时 Java 只 是 简单 地 朋 江 (但 至 少 它 没有 骨 演 我 的 
操作 系统 ) 。 这 已 在 Java 1.1 中 测试 通过 。 


此 时 发 生 的 是 字 串 的 上 自动 类 型 转换 。 当 我 们 使 用 下 述 语句 时 : 
"CrashJava address: " + this 
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西 ， 所 以 它 会 试图 将 this 转 换 成 一 个 字 串 。 转 换 时 调用 的 是 toString(0)， 
后 者 会 产生 一 个 递归 调用 。 若 在 一 个 Vector 内 出 现 这 种 事情 ， 看 起 来 堆 
栈 束 会 溢出 ， 同 时 违例 控制 机 制 根本 没有 机 会 作出 啊 应 。 


在 确实 想 在 这 种 情况 下 打印 出 对 象 的 地 址 ， 解 决 方案 就 是 调用 Object 的 
toString 方 法 。 此 时 就 不 必 加 入 this， 只 需 使 用 super.toString0。 当 然 ， 采 
取 这 种 做 法 也 有 一 个 前 提 : 我 们 必须 从 Object 直接 继承 ， 或 者 没有 一 个 
QAR m J toString YE. 








8.4.2 BitSet 


BitSet 实 际 是 由 “一 进 制 位 ?构成 的 一 个 Vector。 如 采 希 望 高 效率 地 保存 大 
量 “ 开 一 关 ” 信 息 ， 就 应 使 用 BitSet。 它 只 有 从 尺寸 的 角度 看 才 有 意义 ; 
如 果 硕 望 的 高 效率 的 访问 ， 那 么 它 的 速度 会 比 使 用 一 些 固 有 类 型 的 数组 


慢 一 些 。 





此 外 ，BitSet 的 最 小 长 度 是 一 个 长 整数 (Long) 的 长 度 : 64 位 。 这 意味 
着 假如 我 们 准备 保存 比 这 更 小 的 数据 ， 如 8 位 数据 ， 那 么 BitSet 就 显得 浪 
费 了 。 所 以 最 好 创建 自己 的 类 ， 用 它 容 纳 自己 的 标志 位 。 


和 一 个 普通 的 Vector 中 ， 随 我 们 加 入 越 来 越 多 的 元 素 ， 集 合 也 会 自我 脱 
胀 。 在 某 种 程度 上 ，BitSet 也 不 例外 。 也 就 是 说 ， 它 有 时 会 自行 扩展 ， 
有 时 则 不 然 。 而 且 Java 的 1.0 版 本 似乎 在 这 方面 做 得 最 糟 ， 它 的 BitSet 表 
现 十 分 差强人意 〈Javal.1 已 改正 了 这 个 问题 )》。 下 面 这 个 例子 展示 了 
BitSet 是 如 何 运作 的 ， 同 时 演示 了 1.0 版 本 的 错误 ; 


//: Bits.java 





// Demonstration of BitSet 
import java.util.*; 
public class Bits { 
public static void main(String[] args) { 
Random rand = new Random(); 
// Take the LSB of nextInt(): 
byte bt = (byte)rand.nextInt(); 
BitSet bb = new BitSet(); 
for(int i = 7; i >=0; i--) 
if(((1 << i) & bt) != 0) 
bb.set(i); 
else 
bb.clear(i); 
System.out.println("byte value: " + bt); 


printBitSet(bb); 


short st = (short)rand.nextInt(); 


BitSet bs = new BitSet(); 
for(int i = 15; i >=0; i--) 
if(((1 << i) & st) != 0) 
bs.set(i); 
else 
bs.clear(i); 
System.out.println("short value: " + st); 
printBitSet(bs); 
int it = rand.nextInt(); 
BitSet bi = new BitSet(); 
for(int i = 31; i >=0; i--) 
if(((1 << i) & it) != 0) 
bi.set(i); 
else 
bi.clear(i); 
System.out.println("int value: " + it); 
printBitSet(b1); 
// Test bitsets >= 64 bits: 
BitSet b127 = new BitSet(); 
b127.set(127); 
System.out.println("set bit 127: " + b127); 


BitSet b255 = new BitSet(65); 


b255.set(255); 
System.out.println("set bit 255: " + b255); 
BitSet b1023 = new BitSet(512); 
// Without the following, an exception is thrown 
// in the Java 1.0 implementation of BitSet: 
// b1023.set(1023); 
b1023.set(1024); 
System.out.printin("set bit 1023: " + b1023); 
} 
static void printBitSet(BitSet b) { 
System.out.println("bits: " + b); 
String bbits = new String(); 
for(int j = 0; j < b.size() ; j++) 
bbits += (b.get(j) ? "1" : "O"); 
System.out.printin("bit pattern: " + bbits); 
} 
} ///:~ 





随机 数字 生成 器 用 于 创建 一 个 随机 的 byte、short 和 int。 每 一 个 都 会 转换 
成 BitSet 内 相应 的 位 模型 。 此 时 一 切 都 很 正常 ， 因 为 BitSet 是 64 位 的 ， 所 
以 它们 都 不 会 造成 最 终 尺 寸 的 增 大 。 但 在 Java 1.07, —HABitSetK F64 
位 ， 就 会 出 现 一 些 令 人 迷惑 不 解 的 行为 。 假 如 我 们 设置 一 个 只 比 BitSet 
当前 分 配 存储 空间 大 出 1 的 一 个 位 ， 它 能 够 正常 地 扩展 。 但 一 旦 试图 在 
更 高 的 位 置 设置 位 ， 同 时 不 先 接 触 边界 ， 就 会 得 到 一 个 恼人 的 违例 。 这 
正 是 由 于 BitSet 在 Java ”1.0 里 不 能 正确 扩展 造成 的 。 本 例 创 建 了 一 个 512 
位 的 BitSet。 构 建 器 分 配 的 存储 空间 是 位 数 的 两 倍 。 所 以 假如 设置 位 











1024 或 更 高 的 位 ， 同 时 没有 先 设 置 位 1023， 就 会 在 Java ”1.0 里 得 到 一 个 
违例 。 但 幸运 的 是 ， 这 个 问题 已 在 Java 1.1 得 到 了 改正 。 所 以 如 果 是 为 
Java 1.0 写 代码 ， 请 尽量 避免 使 用 BitSet。 


8.4.3 Stack 


Stack 有 时 也 可 以 称 为 "后 入 先 出 ”(LIFO) EA REZ RIER 
里 最 后 < 压 入 ”的 东西 将 是 以 后 第 一 个 弹出 ”的 。 和 其 他 所 有 Java 集 合 一 
样 ， 我 们 压 入 和 弹出 的 都 是 “对 象 "， 所 以 必须 对 自己 弹出 的 东西 进 
行 “造型 ”。 


一 种 很 少见 的 做 法 是 拒绝 使 用 Vector 作为 一 个 Stack 的 基本 构成 元 素 ， 而 
是 从 Vector 里 “继承 ”一 个 Stack。 这 样 一 来 ， 它 束 拥 有 了 一 个 Vector 的 所 
有 特征 及 行为 ， 男 外 加 上 一 些 额外 的 Stack 行 为 。 很 难 判 断 出 设计 者 到 
底 是 明确 想 这 样 做 ， 还 是 属于 一 种 固有 的 设计 。 


下 面 是 一 个 简单 的 堆栈 示例 ， 它 能 读 入 数组 的 每 一 行 ， 同 时 将 其 作为 字 
串 压 入 堆栈 。 


//: Stacks.java 





// Demonstration of Stack Class 
import java.util.*; 
public class Stacks { 
static String[] months = { 
"January", "February", "March", "April", 
"May", "June", "July", "August", "September", 
"October", "November", "December" }; 
public static void main(String[] args) { 
Stack stk = new Stack(); 


for(int i = 0; i < months.length; i++) 


stk.push(months[i] + " "); 
System.out.println("stk = " + stk); 
// Treating a stack as a Vector: 
stk.addElement("The last line"); 
System. out.printin( 

"element 5 = " + stk.elementAt(5)); 
System.out.println("popping elements:"); 
while(!stk.empty()) 

System.out.println(stk.pop()); 

} 
} ///:~ 


months 数 组 的 每 一 行 都 通过 push() 继 承 进 入 堆栈 ， 稍 后 用 pop() 从 堆栈 的 
顶部 将 其 取出 。 要 声明 的 一 点 是 ，Vector 操 作 亦 可 针对 Stack 对 象 进行 。 
这 可 能 是 由 继承 的 特质 决定 的 Stack“ 属 于 ”一 种 Vector。 因 此 ， 能 对 
Vector 进行 的 操作 亦 可 针对 Stack 进 行 ， 例 如 elementAt(0) 方 法 。 





8.4.4 Hashtable 


Vector 允许 我 们 用 一 个 数字 从 一 系列 对 象 中 作出 选择 ， 所 以 它 实 际 是 将 
数字 同 对 象 关联 起 来 了 。 但 假如 我 们 想 根 据 其 他 标准 选择 一 系列 对 象 
We? 堆栈 就 是 这 样 的 一 个 例子 : 它 的 选择 标准 是 “最 后 压 入 堆栈 的 东 
西 ?>。 这 种 “从 一 系列 对 象 中 选择 ”的 概念 亦 可 叫 作 一 个 “映射 “字典 ?或 
者 “关联 数组 ”。 从 概念 上 讲 ， 它 看 起 来 象 一 个 Vector， 但 却 不 是 通过 数 
字 来 查找 对 象 ， 而 是 用 另 一 个 对 象 来 查找 它们 ! 这 通常 都 属于 一 个 程序 
中 的 重要 进程 。 


在 Java 中 ， 这 个 概念 具体 反映 到 抽象 类 Dictionary 身 上 。 该 类 的 接口 是 非 
党 直观 的 size0 告 诉 我 们 其 中 包含 了 多 少 元 素 ; isEmpty0 判 断 是 否 包 含 
了 元 素 〈 是 则 为 tue) ; put(Object key, Object value) 添 加 一 个 值 〈 我 们 

















希望 的 东西 ) ， 并 将 其 同一 个 键 关 联 起 来 〈 想 用 于 搜索 它 的 东西 ) ; 
get(Object key) 获 得 与 某 个 键 对 应 的 值 ， 而 remove(Object Key) 用 于 从 列 
表 中 删除 “ 键 一 值 ? 对 。 还 可 以 使 用 枚 举 技术 : keys(0 产 生 对 键 的 一 个 枚 
举 (Enumeration) ; 而 elements0 产 生 对 所 有 值 的 一 个 枚 举 。 这 便 是 一 
“Dictionary (CFW) 的 全 部 。 


Dictionary 的 实现 过 程 并 不 及 烦 。 下 面 列 出 一 种 简单 的 方法 ， 它 使 用 了 
两 个 Vector， 一 个 用 于 容纳 键 ， 另 一 个 用 来 容纳 值 : 


//: AssocArray.java 


// Simple version of a Dictionary 
import java.util.*; 
public class AssocArray extends Dictionary { 
private Vector keys = new Vector(); 
private Vector values = new Vector(); 
public int size() { return keys.size(); } 
public boolean isEmpty() { 
return keys.isEmpty(); 
} 
public Object put(Object key, Object value) { 
keys.addElement(key); 
values.addElement (value); 
return key; 
} 
public Object get(Object key) { 


int index = keys.indexOf(key); 


// indexOf() Returns -1 if key not found: 
if(index == -1) return null; 

return values.elementAt (index); 

} 

public Object remove(Object key) { 
int index = keys.indexOf (key); 
if(index == -1) return null; 
keys.removeElementAt (index); 
Object returnval = values.elementAt (index) ; 
values.removeElementAt (index) ; 
return returnval; 

} 

public Enumeration keys() { 
return keys.elements(); 

} 

public Enumeration elements() { 
return values.elements(); 

} 

// Test it: 

public static void main(String[] args) { 

AssocArray aa = new AssocArray(); 
for(char c = 'a'; c <= 'z'; C++) 


aa.put(String.valueOf(c), 


String.valueof(c) 
. toUpperCase()); 

char[] ca = { 'a', 'e', 'i', 'o', 'u' }; 

for(int i = 0; i < ca.length; i++) 

System.out.println("Uppercase: " + 
aa.get(String.valueOf(ca[i]))); 
J 
} ///:~ 





在 对 AssocArray 的 定义 中 ， 我 们 注意 到 的 第 一 个 问题 是 它 “ 扩 展 ” 了 字 
典 。 这 意味 着 AssocArray 属 于 Dictionary 的 一 种 类 型 ， 所 以 可 对 其 发 出 与 
Dictionary 一 样 的 请 求 。 如 果 想 生成 自己 的 Dictionary， 而 且 束 在 这 里 进 
行 ， 那 么 要 做 Herat nA 是 填充 位 于 Dictionary 内 的 所 有 方法 (而 且 
必须 和 Ti 都 是 抽象 的 ) 。 


Vector ”key 和 value 通 过 一 个 标准 索引 编写 链接 起 来 。 也 就 是 说 ， 如 果 
用 “roof” 的 一 个 键 以 及 : re ae 假定 我 们 准备 将 一 个 
房子 的 各 部 分 与 它们 的 油漆 颜色 关联 起 来 ， 而 且 AssocArray 里 已 有 100 
个 元 素 ， 那 么 “roof”* 就 会 有 101 个 键 元 系 ， 而 “blue”* 有 101 个 值 元 素 。 而 且 
要 注意 一 下 get()， 假 如 我 们 作为 键 传递 “roof”"， 它 就 会 产生 与 
ne SRS AB S| Sas E BA BAN 


main() 中 进行 的 测试 是 非常 简单 的 ， 它 只 是 将 小 写字 符 转 换 成 大 写字 
oe 更 有 效 的 方式 进行 。 但 它 同 我 们 揭示 出 了 AssocArray 的 
强大 功能 


标准 Java 库 只 包含 Dictionary 的 一 个 变种 ， 名 为 Hashtable( 散 列表 ， 注 释 
©) 。Java 的 散 列 表 具 有 与 AssocArray 相 同 的 接口 〈 因 为 两 者 都 是 从 
Dictionary 继 承 来 的 ) 。 但 有 一 个 方面 却 反 映 出 了 差别 : PAT CR. A 
仔细 想 想 必须 为 一 个 getO 做 的 事情 ， 束 会 发 现在 一 个 Vector 里 搜索 键 的 
速度 要 慢 得 多 。 但 此 时 用 散 列 表 却 可 以 加 快 不 少 速度 。 不 必用 元 长 的 线 




















性 搜索 技术 来 查找 一 个 键 ， 而 是 用 一 个 特殊 的 值 ， 名 为 “ 散 列 码 ”。 散 列 
码 可 以 获取 对 象 中 的 信息 ， 然 后 将 其 转换 成 那个 对 象 “ 相 对 唯一 ”的 整数 
Gnt) 。 所 有 对 象 都 有 一 个 散 列 码 ， 而 hashCode0 是 根 类 Object 的 一 个 
方法 。Hashtable 获 取 对 象 的 hashCode0， 然 后 用 它 快速 查找 键 。 这 样 可 
使 性 能 得 到 大 幅度 提升 〈@ 约 ) 。 散 列表 的 有 具体 工作 原理 已 超出 了 本 书 的 
wH O) 大 家 只 需要 知道 散 列 表 是 一 种 快速 的 “ 字 

W” (Dictionary) 即 可 ， 而 字典 是 一 种 非常 有 用 的 工具 。 


©: 如 计划 使 用 RMI (在 第 15 章 详 述 ) ， 应 注意 将 远程 对 象 置 入 散 列 表 
时 会 过 到 一 个 问题 (参阅 (Core Java》， 作 者 Conrell 和 Horstmann， 
Prentice-Hall 1997 年 出 版 ) 


O: 如 这 种 速度 的 提升 仍然 不 能 满足 你 对 性 能 的 要 求 ， 甚 至 可 以 编写 自 
己 的 散 列 表 例 程 ， 从 而 进一步 加 快 表格 的 检索 过 程 。 这 样 做 可 避免 在 与 
Object 之 间 进 行 造型 的 时 间 延 误 ， 也 可 以 避 开 由 Java 类 库 散 列表 例 程 内 
建 的 同步 过 程 。 


©: 我 的 知道 的 最 佳 参考 读物 是 《Practical Algorithms for 
Programmers》， 作 者 为 Andrew Binstock 和 John Rex，Addison-Wesley 
1995 年 出 版 。 


作为 应 用 散 列 表 的 一 个 例子 ， 可 考虑 用 一 个 程序 来 检验 Java 的 
Math.random() 方 法 的 随机 性 到 抵 如 何 。 在 理想 情况 下 ， 它 应 该 产生 一 系 
列 完 美的 随机 分 布 数字 。 但 为 了 验证 这 一 点 ， 我 们 需要 生成 数量 众多 的 
随机 数字 ， 然 后 计算 落 在 不 同 范围 内 的 数字 多 少 。 散 列表 可 以 极 大 简化 
这 一 工作 ， 因 为 它 能 将 对 象 同 对 象 关 联 起 来 (此 时 是 将 Math.random() 生 
成 的 值 同 那些 值 出 现 的 次 数 关 联 起 来 )》。 如 下 所 示 : 


//: Statistics.java 
































// Simple demonstration of Hashtable 
import java.util.*; 
class Counter { 

int i = 1; 


public String toString() { 


return Integer.toString(i); 


} 
class Statistics { 
public static void main(String[] args) { 
Hashtable ht = new Hashtable(); 
for(int i = 0; i < 10000; i++) { 
// Produce a number between 0 and 20: 
Integer r = 
new Integer((int)(Math.random() * 20)); 
if(ht.containsKey(r)) 
((Counter )ht.get(r)).1i++; 
else 
ht.put(r, new Counter()); 
} 
System.out.printlin(ht); 
} 
} ///:~ 


在 main0 中 ， 每 次 产生 一 个 随机 数字 ， 它 都 会 封装 到 一 个 Integer 对 象 
里 ， 使 句柄 能 够 随同 散 列 表 一 起 使 用 〈 不 可 对 一 个 集合 使 用 基本 数据 类 
型 ， 只 能 使 用 对 象 句 柄 ) 。containKey() 方 法 检查 这 个 键 是 否 已 经 在 集 
合 里 《也 就 是 说 ， 那 个 数字 以 前 发 现 过 吗 ? ) 知已 在 集合 里 ， 则 getO 方 
法 获得 那个 键 关 联 的 值 ， 此 时 是 一 个 Counter (计数 器 〉 对象。 计数 器 
内 的 值 i 随 后 会 增加 1， 表 明 这 个 特定 的 随机 数字 又 出 现 了 一 次 。 


假如 键 以 前 尚未 发 现 过 ， 那 么 方法 putO 仍 然 会 在 散 列 表 内 置 入 一 个 新 
的 “ 键 一 值 * 对 。 在 创建 之 初 ，Counter 会 自己 的 变量 i 自动 初始 化 为 1， 它 
标志 着 该 随机 数字 的 第 一 次 出 现 。 

为 显示 散 列 表 ， 只 需 把 它 简 单 地 打印 出 来 即 可 。Hashtable toString0) 方 法 
能 遍历 所 有 键 一 值 对 ， 并 为 每 一 对 都 调用 toString0)。Integer toString() 是 
事先 定义 好 的 ， 可 看 到 计数 器 使 用 的 toString。 一 次 运行 的 结 末 《添加 
J EBT) SIF: 


{19=526, 18=533, 17=460, 16=513, 15=521, 14=495, 








13=512, 12=483, 11=488, 10=487, 9=514, 8=523, 
7=497, 6=487, 5=480, 4=489, 3=509, 2=503, 1=475, 


0=505} 








大 家 或 许 会 对 Counter 类 是 否 必要 感到 疑惑 ， 它 看 起 来 似乎 根本 没有 封 

装 类 Integer 的 功能 。 为 什么 不 用 int 或 mnteger 呢 ? 事实 上 ， 由 于 所 有 集合 
能 容纳 的 仅 有 对 象 句柄 ， 所 以 根本 不 可 以 使 用 整数 。 学 过 集合 后 ， 封 装 
类 的 概念 对 大 家 来 说 就 可 能 更 容易 理解 了 ， 因 为 不 可 以 将 任何 基本 数据 
类 型 置 入 集合 里 。 人 然而， 我 们 对 Java 封 装 器 能 做 的 唯一 事情 就 是 将 其 初 
始 化 成 一 个 特定 的 值 ， 然 后 读 取 那个 值 。 也 就 是 说 ， 一 旦 封装 器 对 象 已 
经 创建 ， 就 没有 办 法 改变 一 个 值 。 这 使 得 Integer 封 装 器 对 解决 我 们 的 问 
题 蝶 无 意义 ， 所 以 不 得 不 创建 一 个 新 类 ， 用 它 来 满足 自己 的 要 求 。 


1. 创建 “关键 ”类 


在 前 面 的 例子 里 ， 我 们 用 一 个 标准 库 的 类 (Integer〉 作为 Hashtable 的 一 
个 键 使 用 。 作 为 一 个 键 ， 它 能 很 好 地 工作 ， 因 为 它 已 经 具备 正确 运行 的 
所 有 和 条件。 但 在 使 用 散 列表 的 时 候 ， 一 旦 我 们 创建 自己 的 类 作为 键 使 
用 ， 就 会 遇 到 一 个 很 常见 的 问题 。 例 如 ， 假 设 一 套 天 气 预 报 系统 将 
Groundhog (ERMO 对象 匹 配 成 Prediction (WK) 。 这 看 起 来 非常 直 
观 : 我 们 创建 两 个 类 ， 然 后 将 Groundhog 作 为 键 使 用 ， 而 将 Prediction 作 
为 值 使 用 。 如 下 所 示 : 


//: SpringDetector.java 




















// Looks plausible, but doesn't work right. 
import java.util.*; 
class Groundhog { 
int ghNumber; 
Groundhog(int n) { ghNumber = n; } 
} 
class Prediction { 
boolean shadow = Math.random() > 0.5; 
public String toString() { 
if(shadow) 
return "Six more weeks of Winter!"; 
else 
return "Early Spring!"; 
} 
} 


public class SpringDetector { 
public static void main(String[] args) { 
Hashtable ht = new Hashtable(); 
for(int. i = ©; i < 10; i++) 
ht.put(new Groundhog(i), new Prediction()); 
System.out.println("ht = " + ht + "\n"); 


System.out.println( 


"Looking up prediction for groundhog #3:"); 
Groundhog gh = new Groundhog(3); 
if(ht.containsKey(gh) ) 


System.out.printin((Prediction)ht.get(gh)); 


} 
} ///:~ 


每 个 Groundhog 都 具有 一 个 标识 号 码 ， 所 以 赤 了 在 散 列 表 中 查找 一 个 
Prediction， 只 需 指 示 它 “告诉 我 与 Groundhog 号 码 3 相 关 的 Prediction”。 
Prediction 类 包含 了 一 个 布尔 值 ， 用 Math.random() 进 行 初始 化 ， 以 及 一 个 
toString(O) 为 我 们 解释 结果 。 在 main0 中 ， 用 Groundhog 以 及 与 它们 相关 的 
Prediction 填 充 一 个 散 列 表 。 散 列表 被 打印 出 来 ， 以 便 我 们 看 到 它们 确实 
已 被 填充 。 随 后 ， 用 标识 号 码 为 3 的 一 个 Groundhog 碍 找 与 Groundhog #3 
对 应 的 预报 。 


看 起 来 似乎 非常 简单 ， 但 实际 是 不 可 行 的。 问题 在 于 Groundhog 是 从 通 
用 的 Object 根 类 继承 的 〈 知 当初 未 指定 基础 类 ， 则 所 有 类 最 终 都 是 从 
Object 继承 的 ) 。 事 实 上 是 用 Object 的 hashCode0) 方 法 生成 每 个 对 象 的 散 
列 码 ， 而 且 默 认 情 况 下 只 使 用 它 的 对 象 的 地 址 。 所 以 ，Groundhog(3) 的 
第 一 个 实例 并 不 会 产生 与 Groundhog(3) 第 二 个 实例 相等 的 散 列 码 ， 而 我 
们 用 第 二 个 实例 进行 检索 。 


大 家 或 许 认为 此 时 要 做 的 全 部 事情 就 是 正确 地 禾 新 hashCode()。 但 这 样 
做 依然 行 不 能 ， 除 非 再 做 另 一 件 事 情 : 宪 盖 也 属于 Object 一 部 分 的 
equals()。 当 散 列 表 试 图 判断 我 们 的 键 是 否 等 于 表 内 的 某 个 键 时 ， 束 会 
用 到 这 个 方法 。 同 样 地 ， 默 认 的 Object.equalsO 只 是 简单 地 比较 对 象 地 
址 ， 所 以 一 个 Groundhog(3) 并 不 等 于 另 一 个 Groundhog(3)。 


因此 ， 为 了 在 散 列 表 中 将 自己 的 类 作为 键 使 用 ， 必 须 同时 宪 盖 
hashCode() 和 equals()， 就 象 下 面 展示 的 那样 : 


//: SpringDetector2. java 

















// If you create a class that's used as a key in 
// a Hashtable, you must override hashCode( ) 
// and equals(). 
import java.util.*; 
class Groundhog2 { 
int ghNumber; 
Groundhog2(int n) { ghNumber = n; } 
public int hashCode() { return ghNumber; } 
public boolean equals(Object o) { 
return (o instanceof Groundhog2) 


&& (ghNumber == ((Groundhog2)o).ghNumber ) ; 


} 
public class SpringDetector2 { 
public static void main(String[] args) { 
Hashtable ht = new Hashtable(); 
for(int i = ©; i < 10; i++) 
ht.put(new Groundhog2(i),new Prediction()); 
System.out.println("ht = " + ht + "\n"); 
System. out.printin( 
"Looking up prediction for groundhog #3:"); 
Groundhog2 gh = new Groundhog2(3) ; 


if(ht.containsKey(gh) ) 


System.out.printin((Prediction)ht.get(gh)); 


} 
FA ss 


注意 这 段 代 码 使 用 了 来 目前 一 个 例子 的 Prediction， 所 以 
SpringDetector.java 必 须 首 先 编译 ， 人 否则 惑 会 在 试图 编译 
SpringDetector2.java 时 得 到 一 个 编译 期 错误 。 


Groundhog2.hashCode() 将 土 拔 鼠 号 码 作 为 一 个 标识 符 返 回 ( 在 这 个 例子 
中 ， 程 序 员 需 要 保证 没有 两 个 土 拔 鼠 用 同样 的 卫 号 码 并 存 ) 。 为 了 返回 
一 个 独一无二 的 标识 符 ， 并 不 需要 hashCode0，equals() 方 法 必须 能 够 严 
格 判 断 两 个 对 象 是 否 相等 。 


edquals() 方 法 要 进行 两 种 检查 : 检查 对 象 是 否 为 null; 若 不 为 null， 则 继 
续 检 查 是 否 为 Groundhog2 的 一 个 实例 (要 用 到 instanceof 关 键 字 ， 第 11 章 
会 详 加 论述 ) 。 即 使 为 了 继续 执行 equals 中 ， 它 也 应 该 是 一 个 
Groundhog2。 正 如 大 家 看 到 的 那样 ， 这 种 比较 建立 在 实际 ghNumber 的 
基础 上 。 这 一 次 一 旦 我 们 运行 程序 ， 就 会 看 到 它 终 于 产生 了 正确 的 输出 
《许多 Java 库 的 类 都 覆盖 了 hashcode0 和 equals() 方 法 ， 以 便 与 自己 提供 
的 内 容 适 应 ) 。 


2. 属性 : Hashtable 的 一 种 类 型 


在 本 书 的 第 一 个 例子 中 ， 我 们 使 用 了 一 个 名 为 Properties〈 属 性 ) 的 
Hashtable 类 型 。 在 那个 例子 中 ， 下 述 程序 行 : 














Properties p = System.getProperties(); 
p.list(System.out); 


调用 了 一 个 名 为 getProperties() 的 static 方 法 ， 用 于 获得 一 个 特殊 的 
Properties 对 象 ， 对 系统 的 某 些 特征 进行 描述 。list() 属 于 Properties 的 一 个 
方法 ， 可 将 内 容 发 给 我 们 选择 的 任何 流 式 输出 。 也 有 一 个 save() 方 法 ， 
可 用 它 将 属性 列表 写 入 一 个 文件 ， 以 便 日 后 用 load0) 方 法 读 取 。 


尽管 Properties 类 是 从 Hashtable 继 承 的 ， 但 它 也 包含 了 一 个 散 列 表 ， 用 于 


容纳 “默认 ”属性 的 列表 。 所 以 假如 没有 在 主 列表 里 找到 一 个 属性 ， 就 会 
目 动 搜索 默认 属性 。 


Properties 类 亦 可 在 我 们 的 程序 中 使 用 (第 17 章 的 ClassScanner.java 便 是 
一 例 ) 。 在 Java 库 的 用 户 文 档 中 ， 往 往 可 以 找到 更 多 、 更 详细 的 说 明 。 


8.4.5 再 论 枚 举 器 


我 们 现在 可 以 开始 演示 Enumeration (M) 的 真正 威力 : 将 罕 越 一 个 序 
列 的 操作 与 那个 序列 的 基础 结构 分 隔 开 。 在 下 面 的 例子 里 ，PrintData 类 
用 一 个 Enumeration 在 一 个 序列 中 移动 ， 并 为 每 个 对 象 都 调用 toString() 方 
法 。 此 时 创建 了 两 个 不 同类 型 的 集合 : 一 个 Vector 和 一 个 Hashtable。 并 
旦 在 它们 里 面 分 别 填充 Mouse 和 Hamster 对 象 ( 本 章 早 些 时 候 已 定义 了 这 
些 类 ; 注意 必须 先 编 译 HamsterMaze.java 和 WorksAnyway.java， 否 则 下 
面 的 程序 不 能 编译 ) 。 由 于 Enumeration 陷 藏 了 基层 集合 的 结构 ， 所 以 
PrintData 不 知道 或 者 不 关心 Enumeration 来 自 于 什么 类 型 的 集合 : 


//: Enumerators2.java 





// Revisiting Enumerations 
import java.util.*; 
class PrintData { 
static void print(Enumeration e) { 
while(e.hasMoreElements() ) 
System.out.printin( 


e.nextElement().toString()); 


} 
class Enumerators2 { 


public static void main(String[] args) { 


Vector v = new Vector(); 

for(int i = 0; i < 5; i++) 
v.addElement(new Mouse(i)); 

Hashtable h = new Hashtable(); 

for(int i = 0; i < 5; i++) 
h.put(new Integer(i), new Hamster(1)); 

System.out.println("Vector"); 

PrintData.print(v.elements()); 

System.out.println("Hashtable") ; 

PrintData.print(h.elements()); 

} 
} ///:~ 


注意 PrintData.print() 利 用 了 这 些 集 合 中 的 对 象 属于 Object 类 这 一 事实 ， 

所 以 它 调 用 了 toString()。 但 在 解决 自己 的 实际 问题 时 ， 经 和 常 都 要 保证 自 
己 的 Enumeration 罕 越 某 种 特定 类 型 的 集合 。 例 如 ， 可 能 要 求 集合 中 的 所 
有 元 素 都 是 一 个 Shape (几何 形状 ) ， 并 含有 draw() 方 法 。 若 出 现 这 种 情 
况 ， 必 须 从 Enumeration.nextElement(O 返 回 的 Object 进行 下 漳 造 型 ， 以 便 
产生 一 个 Shape。 


8.5 排序 


Java 1.0 和 1.1 库 都 缺少 的 一 样 东 西 是 算术 和 运算， 甚至 没有 最 简单 的 排序 
运算 方法 。 因 此 ， 我 们 最 好 创建 一 个 Vector， 利 用 经 典 的 Quicksort〈 快 
速 排序 ) 方法 对 其 上 自身 进行 排序 。 


编写 通用 的 排序 代码 时 ， 面 临 的 一 个 问题 是 必须 根据 对 象 的 实际 类 型 来 
执行 比较 运算 ， 从 而 实现 正确 的 排序 。 当 然 ， 一 个 办 法 是 为 每 种 不 同 的 
类 型 都 写 一 个 不 同 的 排序 方法 。 然 而 ， 应 认识 到 假若 这 样 做 ， 以 后 增加 
新 类 型 时 便 不 易 实现 代码 的 重复 利用 。 


程序 设计 一 个 主要 的 目标 就 是 “将 发 生变 化 的 东西 同 保持 不 变 的 东西 分 
隔 开 ”。 在 这 里 ， 保 持 不 变 的 代码 是 通用 的 排序 算法 ， 而 每 次 使 用 时 都 
要 变化 的 是 对 象 的 实际 比较 方法 。 因 此 ， 我 们 不 可 将 比较 代码 “ 硬 编 

码 ” 到 多 个 不 同 的 排序 例 程 内 ， 而 是 采用 “回调 ”技术 。 利 用 回调 ， 经 常 
发 生变 化 的 那 部 分 代码 会 封装 到 它 上 自己 的 类 内 ， 而 总 是 保持 相同 的 代码 
则 “回调 ?发 生变 化 的 代码 。 这 样 一 来 ， 不 同 的 对 象 束 可 以 表达 不 同 的 比 
较 方式 ， 同 时 向 它们 传递 相同 的 排序 代码 。 


下 面 这 个 “接口 ”〈Interface) 展示 了 如 何 比 较 两 个 对 象 ， 它 将 那些 “要 发 
生变 化 的 东西 ?封装 在 内 : 


//: Compare.java 





























// Interface for sorting callback: 

package c08; 

interface Compare { 
boolean lessThan(Object lhs, Object rhs); 
boolean lessThanOrEqual(Object lhs, Object rhs); 


Sif es 





对 这 两 种 方法 来 说 ，lhs 代 表 本 次 比较 中 的 “左手 ”对 象 ， 而 rhs 代 表 “ 厂 


PHR. 


可 创建 Vector 的 一 个 子 类 ， 通 过 Compare 实 现 “ 快 速 排序 ”>。 对 于 这 种 算 
法 ， 包 括 它 的 速度 以 及 原理 等 等 在 此 不 具体 说 明 。 欲 知 详情 ， 可 参考 
Binstock 和 Rex 编 著 的 《Practical Algorithms for Programmers) , MH 
Addison-Wesley 于 1995 年 出 版 。 


//: SortVector.java 


// A generic sorting vector 
package c08; 
import java.util.*; 
public class SortVector extends Vector { 
private Compare compare; // To hold the callback 
public SortVector(Compare comp) { 
compare = comp; 
} 
public void sort() { 
quickSort(0, size() - 1); 
} 
private void quickSort(int left, int right) { 
if(right > left) { 
Object o1 = elementAt(right); 
int i = left - 1; 
int j = right; 


while(true) { 


while(compare.lessThan( 
elementAt(++i), 01)) 
while(j > 0) 
if (compare. lessThanOrEqual( 
elementAt(--j), 01)) 
break; // out of while 
if(i >= j) break; 
swap(i, j); 
} 
swap(i , right); 
quickSort(left, i-1); 


quickSort(it+1, right); 


j 


private void swap(int loc1, int loc2) { 
Object tmp = elementAt(loc1); 
setElementAt(elementAt(loc2), loci); 
setElementAt(tmp, loc2); 


} 
Sdn 


现在 ， 大 家 可 以 明日 “回调 ”一 词 的 来 历 ， 这 是 由 于 quickSort() 方 法 “ 往 回 


调用 ”了 Compare 中 的 方法 。 从 中 亦 可 理解 这 种 技术 如 何 生成 通用 的 、 可 
ERAH GFE) 的 代码 。 


为 使 用 SortVector， 必 须 创建 一 个 类 ， 令 其 为 我 们 准备 排序 的 对 象 实现 
Compare。 此 时 内 部 类 并 不 显得 特别 重要 ， 但 对 于 代码 的 组 织 却 是 有 益 
的 。 下 面 是 针对 String 对 象 的 一 个 例子 : 


//: StringSortTest.java 





// Testing the generic sorting Vector 
package c08; 
import java.util.*; 
public class StringSortTest { 
static class StringCompare implements Compare { 
public boolean lessThan(Object 1, Object r) { 
return ((String)1).toLowerCase().compareTo( 
((String)r).toLowerCase()) < 0; 
} 
public boolean 
lessThanOrEqual(Object 1, Object r) { 
return ((String)1).toLowerCase().compareTo( 


((String)r).toLowerCase()) <= 0; 


} 


public static void main(String[] args) { 


SortVector sv = 


new SortVector(new StringCompare()); 
sv.addElement("d"); 
sv.addElement("A"); 
sv.addElement("C"); 
sv.addElement("c"); 
sv.addElement("b"); 
sv.addElement("B"); 
sv.addElement("D"); 
sv.addElement("a"); 
sv.sort(); 
Enumeration e = sv.elements(); 
while(e.hasMoreElements() ) 
System.out.println(e.nextElement()); 
} 
} ///:~ 





内 部 类 是 “静态 ”(Static〉 的 ， 因 为 它 毋 需 连 接 一 个 外 部 类 即 可 工作 。 


大 家 可 以 看 到 ， 一 旦 设置 好 框架 ， 
ve 只 需 简单 地 写 一 个 类 ， 将 “需要 发 生变 化 ”的 东西 封装 进 
去 ， 然 后 将 一 个 对 象 传 给 gone BU Ay. 


比较 时 将 字 串 强制 为 小 写 形式 ， 所 以 大 写 A 会 排列 于 小 写 a 的 和 旁边， 而 不 
会 移动 一 个 完全 不 同 的 地 方 。 然 而， 该 例 也 显示 了 这 种 方法 的 一 个 不 
足 ， 因 为 上 述 测试 代码 按照 出 现 顺序 排列 同一 个 字母 的 大 写 和 小 写 形 
si: AabBcCdD. 但 这 通常 不 是 一 个 大 问题 ， 因 为 经 常 处 理 的 都 是 
更 长 的 字 串 ， 所 以 上 述 效果 不 会 显露 出 来 (Java ”1.2 的 集合 提供 了 排序 
功能 ， 已 解决 了 这 个 问题 ) 。 





继承 Cextends) 在 这 儿 用 于 创建 一 种 新 类 型 的 Vector 也 就 是 说 ， 
SortVector 属 于 一 种 Vector， 并 带 有 一 些 附 加 的 功能 。 继 承 在 这 里 可 发 挥 
很 大 的 作用 ， 但 了 带 来 了 问题 。 它 使 一 些 方法 具有 了 final 属 性 〈 已 在 第 
7 章 讲 述 ) ， 所 以 不 能 覆盖 它们 。 如 果 想 创建 一 个 排 好 序 的 Vector， 令 
其 只 接收 和 生成 String 对 象 ， 就 会 遇 到 有 麻烦。 因为 addElement() 和 
elementAt() 都 具有 final 属 性 ， 而 且 它 们 都 是 我 们 必须 覆盖 的 方法 ， 否 则 
便 无 法 实现 只 能 接收 和 产生 String 对 象 。 

但 在 男 一 方面 ， 请 考虑 采用 “合成 ”方法 : 将 一 个 对 象 置 入 一 个 新 类 的 内 
部 。 此 时 ， 不 是 改写 上 述 代码 来 达到 这 个 目的 ， 而 是 在 新 类 里 简单 地 使 
用 一 个 SortVector。 在 这 种 情况 下 ， 用 于 实现 Compare 接 口 的 内 部 类 就 可 
以 “匿名 ”地 创建 。 如 下 所 示 : 


//: StrSortVector.java 








// Automatically sorted Vector that 
// accepts and produces only Strings 
package c08; 
import java.util.*; 
public class StrSortVector { 
private SortVector v = new SortVector ( 
// Anonymous inner class: 
new Compare() { 
public boolean 
lessThan(Object 1, Object r) { 
return 
((String)1).toLowerCase().compareTo( 


((String)r).toLowerCase()) < 0; 


} 


public boolean 
lessThanOrEqual(Object 1, Object r) { 
return 
((String)1).toLowerCase().compareTo( 


((String)r).toLowerCase()) <= 0; 


); 
private boolean sorted = false; 
public void addElement(String s) { 
v.addElement(s); 
sorted = false; 
} 
public String elementAt(int index) { 
if(!sorted) { 
v.sort(); 
sorted = true; 


} 


return (String)v.elementAt (index); 
} 
public Enumeration elements() { 


if(!sorted) { 


v.sort(); 
sorted = true; 


J 


return v.elements(); 

} 

// Test it: 

public static void main(String[] args) { 
StrSortVector sv = new StrSortVector(); 
sv.addElement("d"); 
sv.addElement("A"); 
sv.addElement("C"); 
sv.addElement("c"); 
sv.addElement("b"); 
sv.addElement("B"); 
sv.addElement("D"); 
sv.addElement("a"); 
Enumeration e = sv.elements(); 
while(e.hasMoreElements() ) 
System.out.println(e.nextElement()); 


} 
VL S/T i= 


这 样 便 可 快速 再 生来 自 SortVector 的 代码 ， 从 而 获得 希望 的 功能 。/ 


大 


yyy 


而 ， 并 不 是 来 自 SortVector 和 Vector 的 所 有 public 方 法 都 能 在 StrSortVector 
中 出 现 。 知 按 这 种 形式 再 生 代 码 ， 可 在 新 类 里 为 包含 类 内 的 每 一 个 方法 
都 生成 一 个 定义 。 当 然 ， 也 可 以 在 刚 开始 时 只 添加 少数 几 个 ， 以 后 根据 
需要 再 添加 更 多 的 。 新 类 的 设计 最 终 会 稳定 下 来 。 


这 种 方法 的 好 处 在 于 它 仍 然 只 接纳 String 对 象 ， 也 只 产生 String 对 象 。 而 
且 相 应 的 检查 是 在 编译 期 间 进行 的 ， 而 非 在 运行 期 。 当 然 ， 只 有 
addFlement0 和 elementAtO 才 具备 这 一 特性 ; elements0 仍 然 会 产生 一 个 
Enumeration 〈 枚 举 ) ， 它 在 编译 期 的 类 型 是 未 定 的 。 当 然 ， 对 
Enumeration 以 及 在 StrSortVector 中 的 类 型 检查 会 照旧 进行 ， 如 果真 的 有 
什么 错误 ， 运 行 期 间 会 简单 地 产生 一 个 违例 。 事 实 上 ， 我 们 在 编译 或 运 
行 期 间 能 保证 一 切 都 正确 无 误 吗 ? (也 就 是 说 ,， “代码 测试 时 也 许 不 能 
保证 ?>， 以 及 * 访 程序 的 用 户 有 可 能 做 一 些 未 经 我 们 测试 的 事情 ?>) 。 尽 
管 存 在 其 他 选择 和 争论 ， 使 用 继承 都 要 容易 得 多 ， 只 是 在 造型 时 让 人 深 
感 不 便 。 同 样 地 ， 一 旦 为 Java 加 入 参数 化 类 型 ， 就 有 望 解决 这 个 问题 。 


大 家 在 这 个 类 中 可 以 看 到 有 一 个 名 为 “sorted” 的 标志 。 每 次 调用 
addElement() 时 ， 都 可 对 Vector 进 行 排序 ， 而 且 将 其 连续 保持 在 一 个 排 好 
序 的 状态 。 但 在 开始 读 取 之 前 ， 人 们 总 是 同一 个 Vector 添加 大 量 元 素 。 
所 以 与 其 在 每 个 addElementO 后 排序 ， 不 如 一 直 等 到 有 人 想 读 取 Vector， 
再 对 其 进行 排序 。 后 者 的 效率 要 蜗 得 多 。 这 种 除非 绝对 必要 ， 否 则 就 不 
采取 行动 的 方法 叫 作 “懒惰 求 值 ”( 还 有 一 种 类 似 的 技术 叫 作 “懒惰 初始 

人 化” 除非 真 的 需要 一 个 字段 值 ， 否 则 不 进行 初始 化 ) 。 





























8.6 通用 集合 库 


通过 本 章 的 学 习 ， 大 家 已 知道 标准 Java 库 提供 了 一 些 特别 有 用 的 集合 ， 
但 距 完 整 意义 的 集合 沿 远 。 除 此 之 外 ， 象 排序 这 样 的 算法 根本 没有 提供 
支持 。C++ 出 色 的 一 个 地 方 就 是 它 的 库 ， 特 别 是 “标准 模板 库 ”(STL) 
提供 了 一 套 相 当 完 整 的 集合 ， 以 及 许多 象 排 序 和 检索 这 样 的 算法 ， 可 以 
非常 方便 地 对 那些 集合 进行 操作 。 有 感 这 一 现状 ， 并 以 这 个 模型 为 基 
础 ，ObjectSpace 公 司 设 计 了 Java 版 本 的 “通用 集合 库 ”( 从 前 叫 作 “Java 通 
用 库 ?， 即 JGL; 但 JGL 这 个 缩写 形式 侵犯 了 Sun 公 司 的 版 权 一 一 尽管 本 
书 仍然 沿用 这 个 简称 ) 。 这 个 库 尽 可 能 遵照 SITE 的 设计 《照顾 到 两 种 语 
言 间 的 差异 ) 。JGL 实 现 了 许多 功能 ， 可 满足 对 一 个 集合 库 的 大 多 数 各 
规 需求 ， 它 与 C++ 的 模板 机 制 非常 相似 。JGL 包 括 相 互 链接 起 来 的 列 
表 、 设 置 、 队 列 、 映 射 、 堆 栈 、 序 列 以 及 反复 器 ， 它 们 的 功能 比 
Enumeration (ÀS) 强 多 了 。 同 时 提供 了 一 套 完整 的 算法 ， 如 检索 和 排 
序 等 。 在 某 些 方面 ，ObjectSpace 的 设计 也 显得 比 Sun 的 库 设计 方案 “ 智 
能 ”一 些 。 举 个 例子 来 说 ，JGL 集 合 中 的 方法 不 会 进入 final 状 态 ， 所 以 很 
容易 继承 和 改写 那些 方法 。 


JGL 已 包括 到 一 些 广 商 发 行 的 Java 套 件 中 ， 而 且 ObjectSpace 公 司 目 己 也 
允许 所 有 用 户 免费 使 用 JGL， 包 括 商 业 性 的 使 用 。 详 细 情 况 和 软件 下 载 
可 访问 http://www.ObjectSpace.com。 与 JGL 配 套 提供 的 联机 文档 做 得 非 
第 好 ， 可 作为 自己 的 一 个 绝 佳 起 点 使 用 。 








8.7 新 集合 


对 我 来 说 ， 集 合 类 属于 最 强大 的 一 种 工具 ， 特 别 适合 在 原创 编程 中 使 
用 。 大 家 可 能 已 感觉 到 我 对 Java 。 ”1.1 提供 的 集合 多 少 有 点 儿 失 望 。 
此 ， 看 到 Java 1.2 对 集合 重新 引起 了 正确 的 注意 后 ， 确 实 令 人 非 第 恰 
快 。 这 个 版 本 的 集合 也 得 到 了 完全 的 重新 设计 《〈 由 Sun 公 司 的 Joshua 
Bloch) 。 我 认为 新 设计 的 集合 是 Java 1.2 中 两 项 最 主要 的 特性 之 一 ( 男 
一 项 是 Swing 库 ， 将 在 第 13 章 叙述 ) ， 因 为 它们 极 大 方便 了 我 们 的 编 
程 ， 也 使 Java 变 成 一 种 更 成 熟 的 编程 系统 。 


有 些 设计 使 得 元 素 间 的 结合 变 得 更 紧密 ， 也 更 容易 让 人 理解 。 例 如 ， 许 
多 名 字 都 变 得 更 短 、 更 明确 了 ， 而 且 更 易 使 用 ， 类 型 同样 如 此 。 有 些 名 
字 进 行 了 修改 ， 更 接近 于 通俗 : 我 感觉 特别 好 的 一 个 是 用 “反复 

ax” CInerator) RÈ JM” (Enumeration) 。 


此 次 重新 设计 也 加 强 了 集合 库 的 功能 。 现 在 新 增 的 行为 包括 链接 列表 、 
队列 以 及 撤消 组 队 〔 即 “ 双 终 点 队列 ”) 。 


集合 库 的 设计 是 相当 困难 的 (会 遇 到 大 量 库 设计 问题 ) 。 在 C++ 中 ， 
STL 用 多 个 不 同 的 类 来 履 盖 基础 。 这 种 做 法 比 起 STL 以 前 是 个 很 大 的 进 
步 ， 那 时 根本 没 做 这 方面 的 考虑 。 但 仍然 没有 很 好 地 转换 到 Java 里 面 。 
结果 就 是 一 大 堆 特 别 容易 混 消 的 类 。 在 另 一 个 极端 ， 我 曾 发 现 一 个 集合 
库 由 单个 类 构成 : colleciton， 它 同时 作为 Vector 和 Hashtable 使 用 。 新 集 
合 库 的 设计 者 则 和 希望 达到 一 种 新 的 平衡 : 实现 人 们 和 希望 从 一 个 成 熟 集合 
库 上 获得 的 完整 功能 ， 同 时 又 要 比 STL 和 其 他 类 似 的 集合 库 更 易学 习 和 
使 用 。 这 样 得 到 的 结果 在 某 些 场合 显得 有 些 古 怪 。 但 和 早期 Java 库 的 一 
些 决 策 不 同 ， 这 些 古怪 之 处 并 非 侦 然 出 现 的 ， 而 是 以 复杂 性 作为 代价 ， 
在 进行 仔细 权衡 之 后 得 到 的 结果 。 这 样 做 也 许 会 延长 人 们 掌握 一 些 库 概 
oe 但 很 快 就 会 发 现 目 己 很 乐于 使 用 那些 新 工具 ， 而 且 变 得 越 来 
离 不 了 它 。 


的 集合 库 考 虑 到 了 “容纳 自己 对 象 * 的 问题 ， 并 将 其 分 割 成 两 个 明确 的 
A: 




















(1) 424 (Collection) : 一 组 单独 的 元 素 ， 通 稼 应 用 了 某 种 规则 。 在 这 
里 ， 一 个 List〈 列 表 ) 必须 按 特定 的 顺序 容纳 元 素 ， 而 一 个 Set〈 集 ) 不 








可 包 合 任何 重复 的 元 素 。 相 反 , “EL” (Bag) 的 概念 未 在 新 的 集合 库 中 
实现 ， 因 为 “列表 ”已 提供 了 类 似 的 功能 


(2) 映射 (Map) : 一 系列 “ 键 一 值 ? 对 〈 这 已 在 散 列 表 身 上 得 到 了 充分 的 
体现 ) 。 从 表面 看 ， 这 似乎 应 该 成 为 一 个 “ 键 一 值 ? 对 的 “集合 ”， 但 假 知 
试图 按 那 种 方式 实现 它 ， 就 会 发 现实 现 过 程 相当 笨拙 。 这 进一步 证 明了 
应 该 分 离 成 单独 的 概念 。 另 一 方面 ， 可 以 方便 地 查看 Map 的 某 个 部 分 。 
只 需 创建 一 个 集合 ， 然 后 用 它 表 示 那 一 部 分 即 可 。 这 样 一 来 ，Map 就 可 
以 返回 自己 键 的 一 个 Set、 一 个 包含 自己 值 的 List 或 省 包含 自己 “ 键 一 
值 ” 对 的 一 个 List。 和 数组 相似 ，Map 可 方便 扩充 到 多 个 “ 维 *”， 考 需 涉及 
任何 新 概念 。 只 需 简 单 地 在 一 个 Map 里 包含 其 他 Map 后 者 叉 可 以 包含 
更 多 的 Map， 以 此 类 推 ) 。 


Collection 和 Map 可 通过 多 种 形式 实现 ， 具 体 由 编程 要 求 决定 。 下 面 列 出 
的 是 一 个 帮助 大 家 理解 的 新 集合 示意 图 : 
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这 张 图 刚 开始 的 时 候 可 能 让 人 有 点 儿 摸 不 着 头脑 ， 但 在 通读 了 本 章 以 
后 ， 相 信 大 家 会 真正 理解 它 实际 只 有 三 个 集合 组 件 : Map，List 和 Set。 











而 且 每 个 组 件 实际 只 有 两 、 三 种 实现 方式 〈 注 释 @OD) ， 而 且 通 常 都 只 有 
一 种 特别 好 的 方式 。 只 要 看 出 了 这 一 点 ， 集 合 就 不 会 再 令 人 生长 。 


©: 写作 本 章 时 ，Java 1.2 尚 处 于 PB 测试 阶段 ， 所 以 这 张 示意 图 没有 包括 
以 后 会 加 入 的 TreeSet。 


虚线 框 代表 “接口 "， 点 线 框 代表 “抽象 ?类 ， 而 实 线 框 代表 普通 (实际 ) 
类 。 点 线 箭 头 表 示 个 特定 的 类 准备 实现 一 个 接口 (在 抽象 关 的 情况 
下 ， 则 是 “部 分 ?实现 一 个 接口 ) 。 双 线 箭 头 表示 一 个 类 可 生成 箭头 指 辐 
的 那个 类 的 对 象 例如 ， 任何 集合 都 可 以 生成 一 个 反复 器 (Iterator) , 
而 一 个 列表 可 以 生成 一 个 ListIterator (以 及 原始 的 反复 器 ， 因为 列表 是 
从 集合 继承 的 ) 。 


致力 于 容纳 对 象 的 接口 是 Collection，List，Set 和 Map。 在 传统 情况 下 ， 
我 们 需要 写 大 量 代 人 码 才能 同 这 些 接口 打交道 。 而 且 为 了 指定 自己 想 使 用 
的 准确 类 型 ， 必 须 在 创建 之 初 进行 设置 。 所 以 可 能 创建 下 面 这 样 的 一 个 
List: 











List x = new LinkedList(); 


当然 ， 也 可 以 诀 定 将 x 作 为 一 kdr (而 不 是 一 个 普通 的 
List〉， 并 用 x 负载 准确 的 类 型 信息 。 使 用 接口 的 好 处 就 是 一 旦 决定 改变 
A SEMEA, 要 做 的 全 部 事情 就 是 在 创建 的 时 候 改 变 它 ， 就 象 下 面 
这 样 : 





List x = new ArrayList(); 
其 余 代码 可 以 保持 原封 不 动 。 


在 类 的 分 级 结构 中 ， 可 看 到 大 量 以 “Abstract” (HHA) 开头 的 类 ， 这 刚 
开始 可 能 会 使 人 感觉 迷惑 。 它 们 实际 上 是 一 些 工 具 ， 用 于 “部 分 ”实现 一 
个 特定 的 接口 。 举 个 例子 来 说 ， 假 如 想 生 成 自己 的 Set， 束 不 是 从 Set 接 
口 开 始 ， 然 后 上 自行 实现 所 有 方法 。 相 反 ， 我 们 可 以 从 AbstractSet 继 承 ， 
只 需 极 少 的 工作 即 可 得 到 目 己 的 新 类 。 尽 管 如 此 ， 新 集合 库 仍 然 包含 了 
足够 的 功能 ， 可 满足 我 们 的 几乎 所 有 需求 。 所 以 考虑 到 我 们 的 目的 ， 可 
忽略 所 有 以 “Abstract” 开 头 的 类 。 


因此 ， 在 观看 这 张 示意 图 时 ， 真 正 需 要 关心 的 只 有 位 于 最 顶部 的 “ 接 














口 ?以 及 普通 《实际 ) 类 一 一 均 用 实 线 方 杠 包围。 通常 需要 生成 实际 类 
的 一 个 对 象 ， 将 其 上 测 造 型 为 对 应 的 接口 。 以 后 即 可 在 代码 的 任何 地 方 
使 用 那个 接口 。 下 面 是 一 个 简单 的 例子 ， 它 用 String 对 象 填充 一 个 集 
合 ， 然 后 打印 出 集合 内 的 每 一 个 元 系 : 


//: SimpleCollection.java 








// A simple example using the new Collections 
package c08.newcollections; 
import java.util.*; 
public class SimpleCollection { 
public static void main(String[] args) { 
Collection c = new ArrayList(); 
for(int i = 0; i < 10; i++) 
c.add(Integer.toString(i)); 
Iterator it = c.iterator(); 
while(it.hasNext() ) 
System.out.println(it.next()); 
} 
} ///:~ 


新 集合 库 的 所 有 代码 示例 都 置 于 子 目 录 newcollections 下， 这 样 便 可 提醒 
A le 1.2 有 效 。 这 样 一 来 ， 我 们 必须 用 下 述 代码 来 
调用 程序 : 


java c08.newcollections.SimpleCollection 


采用 的 语法 与 其 他 程序 是 差不多 的 。 


大 家 可 以 看 到 新 集合 属于 java.util 库 的 一 部 分 ， 所 以 在 使 用 时 不 需要 再 
添加 任何 额外 的 import 语 句 。 


main0 的 第 一 行 创建 了 一 个 ArrayList 对 象 ， 然 后 将 其 上 漳 造 型 成 为 一 个 
集合 。 由 于 这 个 例子 只 使 用 了 Collection 方 法 ， 所 以 从 Collection 继 承 的 
一 个 类 的 任何 对 象 都 可 以 正常 工作 。 但 ArrayList 是 一 个 典型 的 
Collection， 它 代 蔡 了 Vector 的 位 置 。 


显然 ，add() 方 法 的 作用 是 将 一 个 新 元 素 置 入 集合 里 。 然 而 ， 用 户 文 档 谨 
慎 地 指出 add0“ 保 证 这 个 集合 包含 了 指定 的 元 素 *。 这 一 点 是 为 Set 作 铺 
热 的 ， 后 者 只 有 在 元 素 不 存在 的 前 提 下 才 会 真 的 加 入 那个 元 素 。 对 于 
ArrayList 以 及 其 他 任何 形式 的 List，add0O 肯 定 意 味 着 “直接 加 入 ”。 


利用 iterator(0) 方 法 ， 所 有 集合 都 能 生成 一 个 “反复 器 ”(Iterator) 。 反 复 
ae HRs” (Enumeration) ， 是 后 者 的 一 个 替代 物 ， 只 是 : 


(1) ” 它 采 用 了 一 个 历史 上 默认 、 而 且 早 在 OOP 中 得 到 广泛 采纳 的 名 字 
(反复 器 ) 。 


(2) 采用 了 比 Enumeration 更 短 的 名 字 : hasNextO 人 代替 了 
hasMoreElement(), ffijnext(){t## 了 nextElement()。 


(3) 添加 了 一 个 名 为 remove0O 的 新 方法 ， 可 删除 由 Iterator 生 成 的 上 一 个 元 
素 。 所 以 每 次 调用 next() 的 时 候 ， 只 需 调 用 remove() 一 次 。 


在 SimpleCollection.java 中 ， 大 家 可 看 到 创建 了 一 个 反复 器 ， 并 用 它 在 集 
合 里 遍历 ， 打 印 出 每 个 元 素 。 


8.7.1 使 用 Collections 
下 面 这 张 表格 总 结 了 用 一 个 集合 能 做 的 所 有 事情 〈 亦 可 对 Set 和 List 做 同 


样 的 事情 ， 尽 管 List 还 提供 了 一 些 额 外 的 功能 ) 。Map 不 是 从 Collection 
继承 的 ， 所 以 要 单独 对 待 。 




















Boolean add(Object) |/*Ensures that the Collection contains the 


argument. Returns false if it doesn’t add the 
argument. 





Boolean *Adds all the elements in the argument. Returns 
addAll(Collection) true if any elements were added. 
void clear( ) *Removes all the elements in the Collection. 


True if the Collection contains the argument. 


Returns an Iterator that you can use to move 
through the elements in the Collection. 


*If the argument is in the Collection, one instance 
of that element is removed. Returns true if a 
removal occurred. 


*Removes all the elements that are contained in 
removeAll(Collection) the argument. Returns true if any removals 
occurred. 


*Retains only elements that are contained in the 
retainAll(Collection) jjargument (an “intersection” from set theory). 
Returns true if any changes occurred. 


Returns an array containing all the elements in the 
Collection. 





Returns an array containing all the elements in the 
toArray(Object|] a) Collection, whose type is that of the array a rather 
than plain Object (you must cast the array to the 


| [right type). | 


*This is an “optional” method, which means it 
might not be implemented by a particular 


Collection. If not, that method throws an 
UnsupportedOperationException. Exceptions 
will be covered in Chapter 9. 





boolean add(Object) *fRiuERAWNAS SARs. WREAK 
量 ， 就 返回 false〈 假 ) 


boolean A 大 添加 自 变 量 内 的 所 有 元 素 。 如 果 没 有 添加 
元 素 ， 则 返回 true《〈 真 ) 


void clear) 关 删 除 集合 内 的 所 有 元 素 
boolean contains(Object) FREASA TE, MaR A” 


boolean containsAll(Collection) 若 集合 包含 了 自 变量 内 的 所 有 元 素 ， 就 返 
回 “ 真 ” 


boolean isEmpty) FRENATA, WARE A” 
Iterator iterator() 返回 一 个 反复 器 ， 以 用 它 授 历 集合 的 各 元 素 


boolean remove(Object) 火 如 上 自 变 量 在 集合 里 ， 就 删除 那个 元 素 的 一 个 实 
例 。 如 果 已 进行 了 删除 ， 就 返回 “ 真 ” 


boolean removeAll(Collection) 类 删除 自 变 量 里 的 所 有 元 素 。 如 果 已 进行 
了 任何 删除 ， 就 返回 “ 真 ” 


boolean retainAll(Collection) * 只 保留 包含 在 一 个 目 变 量 里 的 元 素 〈 一 个 
PHO AY“ 3052") 如 果 已 进行 了 任何 改变 ， 就 返回 “ 真 ” 


int size() 返回 集合 内 的 元 素数 量 
Object[] toArray() 返回 包含 了 集合 内 所 有 元 素 的 一 个 数组 





























大 这 是 一 个 “可 选 的 ?方法 ， 有 的 集合 可 能 并 未 实现 它 。 若 确实 如 此 ， 访 
方法 就 会 遇 到 一 个 UnsupportedOperatiionException， 即 一 个 “操作 不 文 
持 ” 违 例 ， 详 见 第 9 章 。 


下 面 这 个 例子 同 大 家 演示 了 所 有 方法 。 同 样 地 ， 它 们 只 对 从 集合 继承 的 
东西 有 效 ， 一 个 ArrayList 作 为 一 种 “不 常用 的 分 母 ” 使 用 : 


//: Collection1.java 





// Things you can do with all Collections 
package c08.newcollections; 
import java.util.*; 
public class Collection1 { 
// Fill with 'size' elements, start 
// counting at 'start': 
public static Collection 
fill(Collection c, int start, int size) { 
for(int i = start; i < start + size; i++) 
c.add(Integer.toString(i)); 
return Cc; 
} 
// Default to a "Start" of 0: 
public static Collection 
fill(Collection c, int size) { 


return fill(c, 0, size); 


// Default to 10 elements: 
public static Collection fill(Collection c) { 
return fill(c, 0, 10); 
} 
// Create & upcast to Collection: 
public static Collection newCollection() { 
return fill(new ArrayList()); 
// ArrayList is used for simplicity, but it's 
// only seen as a generic Collection 
// everywhere else in the program. 
} 
// Fill a Collection with a range of values: 
public static Collection 
newCollection(int start, int size) { 
return fill(new ArrayList(), start, size); 
} 
// Moving through a List with an iterator: 
public static void print(Collection c) { 
for(Iterator x = c.iterator(); x.hasNext();) 
System.out.print(x.next() + " "); 
System.out.printin(); 


} 


public static void main(String[] args) { 


Collection c = newCollection(); 

c.add("ten"); 

c.add("eleven"); 

print(c); 

// Make an array from the List: 

Object[] array = c.toArray(); 

// Make a String array from the List: 

String[] str = 

(String[])c.toArray(new String[1]); 

// Find max and min elements; this means 

// different things depending on the way 


// the Comparable interface is implemented: 


System.out.printin("Collections.max(c) = " 
Collections.max(c)); 

System.out.printin("Collections.min(c) = " 
Collections.min(c)); 

// Add a Collection to another Collection 

c.addAll(newCollection()); 

print(c); 

c.remove("3"); // Removes the first one 

print(c); 

c.remove("3"); // Removes the second one 


print(c); 


// Remove all components that are in the 
// argument collection: 
c.removeAll(newCollection()); 
print(c); 
c.addAll(newCollection()); 
print(c); 
// Is an element in this Collection? 
System.out.printin( 
"c.contains(\"4\") = " + c.contains("4")); 
// Is a Collection in this Collection? 
System.out.printin( 
"c,.containsAll(newCollection()) = " + 
c.containsAll(newCollection())); 
Collection c2 = newCollection(5, 3); 
// Keep all the elements that are in both 
// c and c2 (an intersection of sets): 
c.retainAll(c2); 
print(c); 
// Throw away all the elements in c that 
// also appear in c2: 
c.removeAll(c2); 
System.out.println("c.isEmpty() = " + 


c.isEmpty()); 


c = newCollection(); 
print(c); 
c.clear(); // Remove all elements 
System.out.println("after c.clear():"); 
print(c); 
} 
} ///:~ 





通过 第 一 个 方法 ， 我 们 可 用 测试 数据 填充 任何 集合 。 在 当前 这 种 情况 
= 只 是 将 int 转 换 成 String。 第 二 个 方法 将 在 本 章 其 余 的 部 分 经 常 采 


newCollection() 的 两 个 版 本 都 创建 了 ArrayList， 用 于 包含 不 同 的 数据 
集 ， 并 将 它们 作为 集合 对 象 返 回 。 所 以 很 明显 ， 除 了 Collection 接 口 之 
外 ， 不 会 再 用 到 其 他 什么 。 


print() 方 法 也 会 在 本 市 经 常用 到 。 由 于 它 用 一 个 反复 器 (Iterator) 在 一 
个 集合 内 人 表 历 ， 而 任何 集合 都 可 以 产生 这 样 的 一 个 反复 器 ， 所 以 它 适用 
于 List 和 Set， 也 适用 于 由 一 个 Map 生 成 的 Collection 。 


main() 用 简单 的 手段 显示 出 了 集合 内 的 所 有 方法 。 
在 后 续 的 小 节 里 ， 我 们 将 比较 List，Set 和 Map 的 不 同 实现 方案 ， 同 时 指 
出 在 各 种 情况 下 哪 一 种 方案 应 成 为 首选 〈 带 有 星 号 的 那个 ) 。 大 家 会 发 
现 这 里 并 未 包括 一 些 传统 的 类 ， 如 Vector，Stack 以 及 Hashtable 等 。 因 为 
不 管 在 什么 情况 下 ， 新 集合 内 都 有 上 自己 首选 的 类 。 


8.7.2 使 用 Lists 





List Order is the most important feature of a List; it promises to 
(interface) |maintain elements in a particular sequence. List adds a number 
of methods to Collection that allow insertion and removal of 


elements in the middle of a List. (This is recommended only 
for a LinkedList.) A List will produce a ListIterator, and 
using this you can traverse the List in both directions, as well 
as insert and remove elements in the middle of the list (again, 
recommended only for a LinkedList). 


A List backed by an array. Use instead of Vector as a general- 
purpose object holder. Allows rapid random access to elements, 
but is slow when inserting and removing elements from the 
middle of a list. ListIterator should be used only for back-and- 
forth traversal of an ArrayList, but not for inserting and 
removing elements, which is expensive compared to 
LinkedList. 


LinkedList||Provides optimal sequential access, with inexpensive insertions 
and deletions from the middle of the list. Relatively slow for 
random access. (Use ArrayList instead.) Also has addFirst( ), 
addLast( ), getFirst( ), getLast( ), removeFirst( ), and 
removeLast( ) (which are not defined in any interfaces or base 
classes) to allow it to be used as a stack, a queue, and a 
dequeue. 





List GO) ”顺序 是 List 最 重要 的 特性 ， 它 可 保证 元 素 按 照 规 定 的 顺序 
排列 。List 为 Collection 添 加 了 大 量 方法 ， 以 便 我 们 在 List 中 部 插入 和 删 
除 元 素 〈 只 推荐 对 LinkedList 这 样 做 ) 。List 也 会 生成 一 个 

ListIterator( 列表 反复 器 ，〉， 利 用 它 可 在 一 个 列表 里 朝 两 个 方 同 裔 历 ， 
C a ence (同样 地 ， 只 建议 对 LinkedList 这 

羊 做 ) 


ArrayList 炎 ”由 一 个 数组 后 推 得 到 的 List。 作 为 一 个 常规 用 途 的 对 象 容器 
使 用 ， 用 于 替换 原先 的 Vector。 人 允许 我 们 快速 访问 元 素 ， 但 在 从 列表 中 
部 插入 和 删除 元 素 时 ， 速 度 却 嫌 稍 慢 。 一 般 只 应 该 用 Listiterator 对 一 个 
ArrayList 进 行 同 前 和 同 后 过 历 ， 不 要 用 它 删 除 和 插入 元 素 ; 与 
LinkedList 相 比 ， 它 的 效率 要 低 许 多 


LinkedList 提供 优化 的 顺序 访问 性 能 ， 同 时 可 以 高 效率 地 在 列表 中 部 进 

















行 插入 和 删除 操作 。 但 在 进行 随机 访问 时 ， 速 度 却 相当 慢 ， 此 时 应 换 用 
ArrayList。 也 提供 了 addFirst0)，addLastO0，getEFirstD)，getLast()， 
removeFirst() 以 及 removeLast()( 未 在 任何 接口 或 基础 类 中 定义 ) ， 以 便 
将 其 作为 一 个 规格 、 队 列 以 及 一 个 双 同 队列 使 用 


下 面 这 个 例子 中 的 方法 每 个 都 履 盖 了 一 组 不 同 的 行为 : 每 个 列表 都 能 做 
的 事情 CbasicTest()) , WIARE 4837 CiterMotion()) 、 用 一 个 

反复 器 改变 某 些 东 西 CiterManipulation()) 、 体 验 列表 处 理 的 效果 
(testVisual()) 以 及 只 有 LinkedList 才 能 做 的 事情 等 : 


//: List1i.java 








// Things you can do with Lists 
package c08.newcollections; 
import java.util.*; 

public class List1 { 

// Wrap Collection1.fill() for convenience: 
public static List fill(List a) { 

return (List)Collection1.fill(a); 

} 

// You can use an Iterator, just as with a 
// Collection, but you can also use random 
// access with get(): 
public static void print(List a) { 

for(int i = 0; i < a.size(); i++) 
System.out.print(a.get(i) + " "); 


System.out.println(); 


} 


static boolean b; 
static Object 0; 
static int i; 
static Iterator it; 
static ListIterator lit; 
public static void basicTest(List a) { 
a.add(1, "x"); // Add at location 1 
a.add("x"); // Add at end 
// Add a collection: 
a.addAll(fill(new ArrayList())); 
// Add a collection starting at location 3: 
a.addAl1(3, fill(new ArrayList())); 
b = a.contains("1"); // Is it in there? 
// Is the entire collection in there? 
b = a.containsAll(fill(new ArrayList())); 
// Lists allow random access, which is cheap 


// for ArrayList, expensive for LinkedList: 


o a.get(1); // Get object at location 1 


i a.indexOf("1"); // Tell index of object 
// indexOf, starting search at location 2: 
i = a.indexOf("1", 2); 


b 


a.isEmpty(); // Any elements inside? 


it = a.iterator(); // Ordinary Iterator 


= 
H- 
ct 
I 


a.listIterator(); // ListIterator 
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a.listIterator(3); // Start at loc 3 
i = a.lastIndexOf("1"); // Last match 
i = a.lastIndexOf("1", 2); // ...after loc 2 
a.remove(1); // Remove location 1 
a.remove("3"); // Remove this object 
a.set(1, "y"); // Set location 1 to "y" 
// Keep everything that's in the argument 
// (the intersection of the two sets): 
a.retainAll(fill(new ArrayList())); 
// Remove elements in this range: 
a.removeRange(0, 2); 
// Remove everything that's in the argument: 
a.removeAl1(fill(new ArrayList())); 
i = a.size(); // How big is it? 
a.clear(); // Remove all elements 
} 
public static void iterMotion(List a) { 


ListIterator it = a.listIterator(); 


b = it.hasNext(); 
b = it.hasPrevious(); 
o = it.next(); 
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it.nextIndex(); 
o = it.previous(); 
i = it.previousIndex(); 

} 

public static void iterManipulation(List a) { 
ListIterator it = a.listIterator(); 
it.add("47"); 
// Must move to an element after add(): 
it.next(); 
// Remove the element that was just produced: 
it.remove(); 
// Must move to an element after remove(): 
it.next(); 
// Change the element that was just produced: 
it.set("47"); 

} 

public static void testVisual(List a) { 
print(a); 
List b = new ArrayList(); 
fill(b); 
System.out.print("b = "); 
print(b); 


a.addAll(b); 


a.addAl1(fill(new ArrayList())); 

print(a); 

// Shrink the list by removing all the 
// elements beyond the first 1/2 of the list 

System.out.println(a.size()); 

System.out.println(a.size()/2); 

a.removeRange(a.size()/2, a.size()/2 + 2); 

print(a); 

// Insert, remove, and replace elements 
// using a ListIterator: 

ListIterator x = a.listIterator(a.size()/2); 

x.add("one"); 

print(a); 

System.out.println(x.next()); 

x.remove(); 

System.out.println(x.next()); 

x.set("47"); 

print(a); 

// Traverse the list backwards: 

x = a.listIterator(a.size()); 

while(x.hasPrevious()) 

System.out.print(x.previous() + " "); 


System.out.printlin(); 


System.out.println("testVisual finished"); 
} 
// There are some things that only 

// LinkedLists can do: 

public static void testLinkedList() { 
LinkedList 11 = new LinkedList(); 
Collectioni.fill(1ll, 5); 
print(1l); 
// Treat it like a stack, pushing: 
ll.addFirst("one"); 
ll.addFirst("two"); 
print(1l); 
// Like "peeking" at the top of a stack: 
System.out.println(ll.getFirst()); 
// Like popping a stack: 
System.out.println(ll.removeFirst()); 
System.out.println(ll.removeFirst()); 
// Treat it like a queue, pulling elements 

// off the tail end: 
System.out.println(1ll.removeLast()); 
// With the above operations, it's a dequeue! 


print(1l); 


public static void main(String args[]) { 
// Make and fill a new list each time: 
basicTest(fill(new LinkedList())); 
basicTest(fill(new ArrayList())); 
iterMotion(fill(new LinkedList())); 
iterMotion(fill(new ArrayList())); 
iterManipulation(fill(new LinkedList())); 
iterManipulation(fill(new ArrayList())); 
testVisual(fill(new LinkedList())); 
testLinkedList(); 


} 
} ///:~ 


在 basicTest() 和 iterMotiion(O) 中 ， 只 是 简单 地 发 出 调用 ， 以 便 揭 示 出 正确 
的 语法 。 而 且 尺 管 捕获 了 返回 值 ， 但 是 并 未 使 用 它 。 在 某 些 情况 下 ,之 
所 以 不 捕获 返回 值 ， 是 由 于 它们 没有 什么 特别 的 用 处 。 在 正式 使 用 它们 
下 应 仔细 研究 一 下 自己 的 联机 文档 ， 掌 握 这 些 方 法 完整 、 正 确 的 用 
YE 

8.7.3 使 用 Sets 

Set 拥 有 与 Collection 完 全 相同 的 接口 ， 所 以 和 两 种 不 同 的 List 不 同 ， 它 没 
有 什么 额外 的 功能 。 相 反 ，Set 完 全 束 是 一 个 Collection， 只 是 具有 不 同 
的 行为 (这 是 实例 和 多 形 性 最 理想 的 应 用 : 用 于 表达 不 同 的 行为 ) 。 在 


这 里 ， 一 个 Set 只 允许 每 个 对 象 存在 一 个 实例 〈 正 如 大 家 以 后 会 看 到 的 
那样 ， 一 个 对 象 的 “ 值 ?的 构成 是 相当 复杂 的 ) 。 


Set 区 element that you add to the Set must be unique; ews 














(interface)llthe Set doesn’t add the duplicate element. Objects added to a Set 
must define equals( ) to establish object uniqueness. Set has 
exactly the same interface as Collection. The Set interface does 
not guarantee it will maintain its elements in any particular order. 





HashSet* |For Sets where fast lookup time is important. Objects must also 
define hashCode( ). 

TreeSet {An ordered Set backed by a red-black tree. This way, you can 
extract an ordered sequence from a Set. 


Set GZO) AMESA ICA MY AE TCI; 否则 Set 就 不 会 
添加 重复 的 元 素 。 添 加 到 Set 里 的 对 象 必 须 定义 equals()， 从 而 建立 对 象 
的 唯一 性 。Set 拥 有 与 Collection 完 全 相同 的 接口 。 一 个 Set 不 能 保证 自己 
可 按 任 何 特定 的 顺序 维持 自己 的 元 素 


HashSet * 用 于 除非 常 小 的 以 外 的 所 有 Set。 对 象 也 必须 定义 hashCode() 


ArraySet ”由 一 个 数组 后 推 得 到 的 Set。 面 同 非常 小 的 Set 设 计 ， 特 别 是 那 
些 需 要 频繁 创建 和 删除 的 。 对 于 小 Set， 与 HashSet 相 比 ，ArraySet 创 建 
和 反复 所 需 付出 的 代价 都 要 小 得 多 。 但 随 着 Set 的 增 大 ， 它 的 性 能 也 会 
大 打折 扣 。 不 需要 HashCode() 


TreeSet 由 一 个 “ 红 黑 树 ” 后 推 得 到 的 顺序 Set CRO) 。 这 样 一 来 ， 我 
们 就 可 以 从 一 个 Set 里 提 到 一 个 顺序 集合 

O: 直至 本 书写 作 的 时 候 ，TreeSet 仍 然 只 是 宣布 ， 尚 未 正式 实现 。 所 以 
这 里 没有 提供 使 用 TreeSet 的 例子 。 

下 面 这 个 例子 并 没有 列 出 用 一 个 Set 能 够 做 的 全 部 事情 ， 因 为 接口 与 
Collection 是 相同 的 ， 前 例 已 经 练习 过 了 。 相 反 ， 我 们 要 例 示 的 重点 在 于 
使 一 个 Set 独 一 无 二 的 行为 : 


//: Seti.java 


= 








// Things you can do with Sets 


package c08.newcollections; 
import java.util.*; 
public class Set1 { 
public static void testVisual(Set a) { 
Collection1i.fill(a)/; 
Collection1i.fill(a); 
Collection1i.fill(a); 
Collectioni.print(a); // No duplicates! 
// Add another set to this one: 
a.addAll(a)j; 
a.add("one"); 
a.add("one"); 
a.add("one"); 
Collection1.print(a); 


// Look something up: 


System.out.printin("a.contains(\"one\"): 


a.contains("one")); 


} 
public static void main(String[] args) { 
testVisual(new HashSet()); 


testVisual(new TreeSet()); 


de 





重复 的 值 被 添加 到 Set， 但 在 打印 的 时 候 ， 我 们 会 发 现 Set 只 接受 每 个 值 
的 一 个 实例 。 


运行 这 个 程序 时 ， 会 注意 到 由 HashSset 维 持 的 顺序 与 ArraySet 是 不 同 的 。 
这 是 由 于 它们 采用 了 不 同 的 方法 来 保存 元 素 ， 以 便 它们 以 后 的 定位 。 
ArraySet 保 持 着 它们 的 顺序 状态 ， 而 HashSet 使 用 一 个 散 列 函数 ， 这 是 特 
别 为 快速 检索 设计 的 ) 。 创 建 自己 的 类 型 时 ， 一 定 要 注意 Set 需 要 通过 
一 种 方式 来 维持 一 种 存储 顺序 ， 束 和 象 本 章 早 些 时 候 展 示 

的 “groundhog”( 土 拔 鼠 )〉 例子 那样 。 下 面 是 一 个 例子 : 


//: Set2.java 

















// Putting your own type in a Set 
package c08.newcollections; 
import java.util.*; 
class MyType implements Comparable { 
private int 1; 
public MyType(int n) { 1 =n; } 
public boolean equals(Object o) { 
return 
(o instanceof MyType) 
&& (i == ((MyType)o).i); 
} 
public int hashCode() { return i; } 
public String toString() { returni+" "; } 


public int compareTo(Object o) { 


int i2 = ((MyType) o).i; 


return (12 <i? -1 : (12 = i ? O : 1)); 


} 


public class Set2 { 

public static Set fill(Set a, int size) { 

for(int i = 0; i < size; i++) 
a.add(new MyType(i)); 

return a; 

} 

public static Set fill(Set a) { 
return fill(a, 10); 

} 

public static void test(Set a) { 
fill(a); 
fill(a); // Try to add duplicates 
fill(a); 
a.addAll(fill(new TreeSet())); 
System.out.println(a); 

} 

public static void main(String[] args) { 
test(new HashSet()); 


test(new TreeSet()); 


} 
} ///:~ 


对 equals() 及 hashCode() 的 定义 遵照 “groundhog” 例 子 已 经 给 出 的 形式 。 在 
两 种 情况 下 都 必须 定义 一 个 equals0。 但 只 有 要 把 类 置 入 一 个 HashSet 的 
前 提 下 ， 才 有 必要 使 用 hashCode() 
通常 应 先 选 择 作为 一 个 Set 实 现 。 


8.7.4 使 用 Maps 


Map Maintains key-value associations (pairs), so you can look up a 
interface) livalue using a key. 


HashMap*iImplementation based on a hash table. (Use this instead of 
Hashtable.) Provides constant-time performance for inserting 
and locating pairs. Performance can be adjusted via 
constructors that allow you to set the capacity and load factor 








这 种 情况 是 完全 有 可 能 的 ， 因 为 





of the hash table. 


TreeMap {Implementation based on a red-black tree. When you view the 
keys or the pairs, they will be in sorted order (determined by 
Comparable or Comparator, discussed later). The point of a 
TreeMap is that you get the results in sorted order. TreeMap 


is the only Map with the subMap() method, which allows you 
to return a portion of the tree. 





Map (ZO) 维持 “ 键 一 值 ? 对 应 关系 〈 对 ) ， 以 便 通 过 一 个 键 得 找 相 应 
的 值 


HashMap * 基于 一 个 散 列 表 实 现 《〈 用 它 代 蔡 Hashtable) 。 针 对 “ 键 一 
值 ” 对 的 插入 和 检索 ， 这 种 形式 具有 最 稳定 的 性 能 。 可 通过 构建 器 对 这 
一 性 能 进行 调整 ， 以 便 设 置 散 列 表 的 “能 力 ” 和 “装载 因子 ” 


ArrayMap 由 一 个 ArrayList 后 推 得 到 的 Map。 对 反复 的 顺序 提供 了 精确 的 











控制 。 面 癌 非 常 小 的 Map 设 计 ， 特 别 是 那些 需要 经 常 创 建 和 删除 的 。 对 
于 非常 小 的 Map， 创 建 和 反复 所 付出 的 代价 要 比 HashMap 低 得 多 。 但 在 
Map 变 大 以 后 ， 性 能 也 会 相应 地 大 幅度 降低 


TreeMap 在 一 个 “ 红 一 黑 ” 树 的 基础 上 实现 。 查 看 键 或 者 “ 键 一 值 ” 对 时 ， 
它们 会 按 固 定 的 顺序 排列 (取决 于 Comparable 或 Comparator， 稍 后 即 会 
讲 到 ) 。TreeMap 最 大 的 好 处 就 是 我 们 得 到 的 是 已 排 好 序 的 结果 。 

TreeMap 是 含有 subMap() 方 法 的 唯一 一 种 Map， 利 用 它 可 以 返回 树 的 一 


部 分 


下 例 包含 了 两 套 测 试 数据 以 及 一 个 fl0 方 法 ， 利 用 该 方法 可 以 用 任何 两 
维 数组 “由 Object 构成 ) 填充 任何 Map。 这 些 工具 也 会 在 其 他 Map 例 子 
中 用 到 。 


//: Map1,java 


// Things you can do with Maps 
package c08.newcollections; 
import java.util.*; 
public class Mapi { 

public final static String[][] testData1 = { 
"Happy", "Cheerful disposition" }, 
"Sleepy", "Prefers dark, quiet places" }, 
"Grumpy", "Needs to work on attitude" }, 
"Doc", "Fantasizes about advanced degree"}, 
"Dopey", "'A' for effort" }, 


"Sneezy", "Struggles with allergies" }, 
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"Bashful", "Needs self-esteem workshop"}, 


}; 


public final static String[][] testData2 = { 
{ "Belligerent", "Disruptive influence" }, 
{ "Lazy", "Motivational problems" }, 
{ "Comatose", "Excellent behavior" } 

}; 

public static Map fill(Map m, Object[][] o) { 
for(int i = 0; i < o.length; i++) 

m.put(o[i][0], of[i][1]); 

return m; 

} 

// Producing a Set of the keys: 

public static void printKeys(Map m) { 

System.out.print("Size = " + m.size() +", "); 
System.out.print("Keys: "); 
Collection1.print(m.keySet()); 

} 

// Producing a Collection of the values: 

public static void printValues(Map m) { 

System.out.print("Values: "); 
Collection1.print(m.values()); 

} 

// Iterating through Map.Entry objects (pairs): 


public static void print(Map m) { 


Collection entries = m.entries(); 
Iterator it = entries.iterator(); 
while(it.hasNext()) { 
Map.Entry e = (Map.Entry)it.next(); 
System.out.println("Key = " + e.getKey() + 


", Value = " + e.getValue()); 


j 


public static void test(Map m) { 

fill(m, testData1); 

// Map has 'Set' behavior for keys: 

fill(m, testData1); 

printKeys(m); 

printValues(m); 

print(m); 

String key = testDatai[4][0]; 

String value = testDatai[4][1]; 

System.out.println("m.containsKey(\"" + key + 
"\"): " + m.containsKey(key)); 

System.out.printin("m.get(\"" + key + "\"): " 
+ m.get(key)); 

System.out.println("m.containsValue(\"" 


+ value + "\"): "+ 


m.containsValue(value) ); 
Map m2 = fill(new TreeMap(), testData2); 
m.putAl1(m2); 
printKeys(m); 
m.remove(testData2[0][0]); 
printKeys(m); 
m.clear(); 
System.out.println("m.isEmpty(): " 
+ m.isEmpty()); 
fill(m, testData1); 
// Operations on the Set change the Map: 
m.keySet().removeAll(m.keySet()); 
System.out.println("m.isEmpty(): " 
+ m.isEmpty()); 
} 
public static void main(String args[]) { 
System.out.println("Testing HashMap"); 
test(new HashMap()); 


System.out.println("Testing TreeMap"); 


test(new TreeMap()); 


} ///:~ 


printKeys()，printValues() 以 及 print() 方 法 并 不 只 是 有 用 的 工具 ， 它 们 也 
清楚 地 揭示 了 一 个 Map 的 Collection“ 景 象 ” 的 产生 过 程 。keySet0 方 法 会 
产生 一 个 Set， 它 由 Map 中 的 键 后 推 得 来 。 在 这 儿 ， 它 只 被 当 作 一 个 
Collection 对 待 。values() 也 得 到 了 类 似 的 对 待 ， 它 的 作用 是 产生 一 个 
List， 其 中 包含 了 Map 中 的 所 有 值 〈 注 意 键 必须 是 独一无二 的 ， 而 值 可 
以 有 重复 ) 。 由 于 这 些 Collection 是 由 Map 后 推 得 到 的 ， 所 以 一 个 
Collection 中 的 任何 改变 都 会 在 相应 的 Map 中 反映 出 来 。 


printO 方 法 的 作用 是 收集 由 entries 产 生 的 Iterator〈 反 复 器 ) ， 并 用 它 同 时 
打印 出 每 个 “ 键 一 值 ? 对 的 键 和 值 。 程 序 剩余 的 部 分 提供 了 每 种 Map 操 作 
的 简单 示例 ， 并 对 每 种 类 型 的 Map 进 行 了 测试 。 


当 创 建 自己 的 类 ， 将 其 作为 Map 中 的 一 个 键 使 用 时 ， 必 须 注意 到 和 以 前 
的 Set 相 同 的 问题 。 


8.7.5 决定 实施 方案 


从 早 些 时 候 的 那 幅 示意 图 可 以 看 出 ， 实 际 上 只 有 三 个 集合 组 件 : Map, 
List 和 Set。 而 且 每 个 接口 只 有 两 种 或 三 种 实施 方 采 。 奉 需 使 用 由 一 个 特 
定 的 接口 提供 的 功能 ， 如 何 才能 决定 到 底 采 取 哪 一 种 方案 呢 ? 


为 理解 这 个 问题 ， 必 须 认识 到 每 种 不 同 的 实施 方案 都 有 自己 的 特点 、 优 
点 和 缺点 。 比 如 在 那 张 示 意图 中 ， 可 以 看 到 Hashtable，Vector 和 Stack 
的 “特点 ”是 它们 都 属于 “传统 "类 ， 所 以 不 会 干扰 原 有 的 代码 。 但 在 男 一 
方面 ， 应 尽量 避免 为 新 的 《Java 1.2) 代码 使 用 它们 。 


其 他 集合 间 的 差异 通常 都 可 归纳 为 它们 具体 是 由 什么 “后 推 " 的 。 换 言 
之 ， 取 决 于 物理 音义 上 用 于 实施 目标 接口 的 数据 结构 是 什么 。 例 如 ， 
ArrayList，LinkedList 以 及 Vector〈 大 致 等 价 于 ArrayList) 都 实现 了 List 
接口 ， 所 以 无 论 选 用 哪 一 个 ， 我 们 的 程序 都 会 得 到 类 似 的 结果 。 然 而 ， 
ArrayList (VAR Vector) 是 由 一 个 数组 后 推 得 到 的 ;而 LinkedList 是 根据 
常规 的 双重 链接 列表 方式 实现 的 ， 因 为 每 个 单独 的 对 象 都 包含 了 数据 以 
及 指 回 列表 内 前 后 元 素 的 句柄 。 正 是 由 于 这 个 原因 ， 假 如 想 在 一 个 列表 
中 部 进行 大 量 插 入 和 删除 操作 ， 那 么 LinkedList 无 疑 是 最 恰当 的 选择 
CLinkedList 还 有 一 些 额外 的 功能 ， 建 立 于 AbstractSequentialList 中 ) 。 
若非 如 此 ， 就 情愿 选择 ArrayList， 它 的 速度 可 能 要 快 一 些 。 


作为 另 一 个 例子 ，Set 既 可 作为 一 个 ArraySet 实 现 ， 亦 可 作为 HashSet 实 























现 。ArraySet 是 由 一 个 ArrayList 后 推 得 到 的 ， 设 计 成 只 支持 少量 元 素 ， 
特别 适合 要 求 创建 和 删除 大 量 Set 对 象 的 场合 使 用 。 然 而 ， 一 旦 需要 在 
自己 的 Set 中 容纳 大 量 元 素 ，ArraySet 的 性 能 就 会 大 打折 扣 。 写 一 个 需要 
Set 的 程序 时 ， 应 默认 选择 HashSet。 而 且 只 有 在 某 些 特殊 情况 下 (对 性 
能 的 提升 有 迫切 的 需求 ) ， 才 应 切换 到 ArraySet。 


1. 决定 使 用 何 种 List 

为 体会 各 种 List 实 施 方案 间 的 差 蜡 ， 最 简便 的 方法 束 是 进行 一 次 性 能 测 
验 。 下 述 代码 的 作用 是 建立 一 个 内 部 基础 类 ， 将 其 作为 一 个 测试 床 使 
用 。 然 后 为 每 次 测验 都 创建 一 个 匿名 内 部 类 。 每 个 这 样 的 内 部 类 都 由 一 
个 test() 方 法 调用 。 利 用 这 种 方法 ， 可 以 方便 添加 和 删除 测试 项 目 。 


//: ListPerformance, java 








// Demonstrates performance differences in Lists 
package c08.newcollections; 
import java.util.*; 
public class ListPerformance { 
private static final int REPS = 100; 
private abstract static class Tester { 
String name; 
int size; // Test quantity 
Tester(String name, int size) { 
this.name = name; 
this.size = size; 
} 


abstract void test(List a); 


private static Tester[] tests = { 
new Tester("get", 300) { 
void test(List a) { 
for(int i = 0; i < REPS; i++) { 
for(int j = 0; j < a.size(); j++) 


a.get(j); 


ty 


new Tester("iteration", 300) { 
void test(List a) { 
for(int i = 0; i < REPS; i++) { 
Iterator it = a.iterator(); 
while(it.hasNext()) 


it.next(); 


ty 
new Tester("insert", 1000) { 


void test(List a) { 


int half a.size()/2; 
String s = "test"; 


ListIterator it = a.listIterator(half); 


for(int i = 0; i < size * 10; i++) 


it.add(s); 


ty 


new Tester("remove", 5000) { 
void test(List a) { 
ListIterator it = a.listIterator(3); 
while(it.hasNext()) { 
it.next(); 


it.remove(); 


ty 
}; 
public static void test(List a) { 
// A trick to print out the class name: 
System.out.println("Testing " + 
a.getClass().getName()); 
for(int i = 0; i < tests.length; i++) { 
Collectioni.fill(a, tests[i].size); 
System.out.print(tests[i].name); 
long t1 = System.currentTimeMillis(); 


tests[i].test(a); 


long t2 = System.currentTimeMillis(); 


System.out.printin(": " + (t2 - t1)); 


} 
public static void main(String[] args) { 
test(new ArrayList()); 
test(new LinkedList()); 
} 
} ///:~ 





内 部 类 Tester 是 一 个 抽象 类 ， 用 于 为 特定 的 测试 提供 一 个 基础 关 。 它 包 
含 了 一 个 要 在 测试 开始 时 打印 的 字 串 、 一 个 用 于 计算 测试 次 数 或 元 素数 
量 的 size 参 数 、 用 于 初始 化 字段 的 一 个 构建 器 以 及 一 个 抽象 方法 test()。 
testO 做 的 是 最 实际 的 测试 工作 。 各 种 类 型 的 测试 都 集中 到 一 个 地 方 : 
tests 数 组 。 我 们 用 继承 于 Tester 的 不 同 匿名 内 部 类 来 初始 化 该 数组 。 为 
添加 或 删除 一 个 测试 项 目 ， 只 二 在 数组 里 简单 地 添加 或 移 去 一 个 内 部 类 
定义 即 可 ， 其 他 所 有 工作 都 是 目 动 进行 的 。 


首先 用 元 素 填 充 传 递 给 test() 的 List， 然 后 对 tests 数 组 中 的 测试 计时 。 由 
于 测试 用 机 器 的 不 同 ， 结 果 当 然 也 会 有 所 区 别 。 这 个 程序 的 宗 则 是 揭示 
出 不 同 集合 类 型 的 相对 性 能 比较 。 下 面 是 某 一 次 运行 得 到 的 结 

类 型 获取 反复 插入 删除 


ArrayList 110 270 1920 4780 














LinkedList 1870 7580 170 110 


可 以 看 出 ， 在 ArrayList 中 进行 随机 访问 〈 即 getO0) 以 及 循环 反复 是 最 划 
得 来 的 ， 但 对 于 LinkedList 却 是 一 个 不 小 的 开销 。 但 另 一 方面 ， 在 列表 
中 部 进行 插入 和 删除 操作 对 于 LinkedList 来 说 却 比 ArrayList 划 算得 多 。 
我 们 最 好 的 做 法 也 许 是 先 选择 一 个 ArrayList 作 为 自己 的 默认 起 点 。 以 后 





苛 发 现 由 于 大 量 的 插入 和 删除 造成 了 性 能 的 降低 ， 再 考 上 处 换 成 
LinkedList 不 迟 。 


2. 决定 使 用 何 种 Set 


可 在 ArraySet 以 及 HashSet 间 作出 选择 ， 有 基体 取 雇 于 Set 的 大 小 《如 有 果 需 
要 从 一 个 Set 中 获得 一 个 顺序 列表 ， 请 用 TreeSet; 注释 @)) 。 下 面 这 个 
测试 程序 将 有 助 于 大 家 作出 这 方面 的 抉择 : 


//: SetPerformance.java 


package c08.newcollections; 
import java.util.*; 
public class SetPerformance { 
private static final int REPS = 200; 
private abstract static class Tester { 
String name; 
Tester(String name) { this.name = name; } 
abstract void test(Set s, int size); 
} 
private static Tester[] tests = { 
new Tester("add") { 
void test(Set s, int size) { 
for(int i = 0; i < REPS; i++) { 
s.clear(); 


Collectioni.fill(s, size); 


ty 
new Tester("contains") { 
void test(Set s, int size) { 
for(int i = 0; i < REPS; i++) 
for(int j = 0; j < size; j++) 


s.contains(Integer.toString(j)); 


ty 


new Tester("iteration") { 
void test(Set s, int size) { 
for(int i = 0; i < REPS * 10; i++) { 
Iterator it = s.iterator(); 
while(it.hasNext()) 


it.next(); 


ty 

}; 

public static void test(Set s, int size) { 
// A trick to print out the class name: 
System.out.println("Testing " + 


s.getClass().getName() + " size " + size); 


Collectioni.fill(s, 


for(int i = 0; i< 


size); 


tests.length; i++) { 


System.out.print(tests[i].name); 


long t1 = System. 
tests[i].test(s, 
long t2 = System. 


System.out.printin(": " 


((double)(t2 - 


// Small: 
test(new TreeSet(), 
test(new HashSet(), 
// Medium: 
test(new TreeSet(), 
test(new HashSet(), 
// Large: 
test(new HashSet(), 


test(new TreeSet(), 


Fj is 


currentTimeMillis(); 
size); 
currentTimeMillis(); 
+ 


t1)/(double)size)); 


public static void main(String[] args) { 


10); 


10); 


100); 


100); 


1000); 


1000); 


©: TreeSet 在 本 书写 作 时 尚未 成 为 一 个 正式 的 特性 ， 但 在 这 个 例子 中 可 
以 很 轻松 地 为 其 添加 一 个 测试 。 





最 后 对 ArraySet 的 测试 只 有 500 个 元 素 ， 而 不 是 1000 个 ， 因 为 它 太 慢 了 。 
类 型 测试 大 小 添加 包含 反复 





eee ee HashSet 显 然 要 比 ArraySet 出 色 得 多 ， 而 
且 性 能 明显 与 元 素 的 多 奔 关系 不 大 。 一 般 编 写 程 序 的 时 候 ， 几 乎 永远 用 
不 着 使 用 ArraySet。 


3. 决定 使 用 何 种 Map 


选择 不 同 的 Map 实 施 方案 时 ， 注 意 Map 的 大 小 对 于 性 能 的 影响 是 最 大 
的 ， 下 面 这 个 测试 程序 清楚 地 阐 示 了 这 一 点 : 


//: MapPerformance.java 





// Demonstrates performance differences in Maps 


package c08.newcollections; 


import java.util.*; 
public class MapPerformance { 
private static final int REPS = 200; 
public static Map fill(Map m, int size) { 
for(int i = 0; i < size; i++) { 
String x = Integer.toString(1); 
m.put(x, x); 
} 
return m; 
} 
private abstract static class Tester { 
String name; 
Tester(String name) { this.name = name; } 
abstract void test(Map m, int size); 
} 
private static Tester[] tests = { 
new Tester("put") { 
void test(Map m, int size) { 
for(int i = 0; i < REPS; i++) { 
m.clear(); 


fill(m, size); 


ty 
new Tester("get") { 
void test(Map m, int size) { 
for(int i = 0; i < REPS; i++) 
for(int j = 0; j < size; j++) 


m.get(Integer.toString(j)); 


ty 
new Tester("iteration") { 
void test(Map m, int size) { 
for(int i = 0; i < REPS * 10; i++) { 
Iterator it = m.entries().iterator(); 
while(it.hasNext()) 


it.next(); 


ty 
}; 
public static void test(Map m, int size) { 
// A trick to print out the class name: 
System.out.println("Testing " + 
m.getClass().getName() + " size " + size); 


fill(m, size); 


for(int i = 0; i < tests.length; i++) { 
System.out.print(tests[i].name); 
long t1 = System.currentTimeMillis(); 
tests[i].test(m, size); 
long t2 = System.currentTimeMillis(); 
System.out.printin(": " + 


((double)(t2 - t1)/(double)size)); 


} 

public static void main(String[] args) { 
// Small: 
test(new Hashtable(), 10); 
test(new HashMap(), 10); 
test(new TreeMap(), 10); 
// Medium: 
test(new Hashtable(), 100); 
test(new HashMap(), 100); 
test(new TreeMap(), 100); 
// Large: 
test(new HashMap(), 1000); 
test(new Hashtable(), 1000); 


test(new TreeMap(), 1000); 


} ///:~ 


haa 问题 ， 所 以 程序 的 计时 测试 按 Map 的 大 小 
(或 容量 ) 来 分 割 时 间 ， 以 便 得 到 令 人 信服 的 测试 结果 。 下 面 列 出 一 系 
列 结果 《在 你 的 机 器 上 可 能 不 同 ) : 


类 型 测试 大 小 置 入 取出 反复 
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即使 大 小 为 10，ArrayMap 的 性 能 也 要 比 HashMap 差 除 反 复 循环 时 以 
外 。 而 在 使 用 Map 时 ， 反 复 的 作用 通常 并 不 重要 (get() 通 常 是 我 们 时 间 
花 得 最 多 的 地 方 )。TreeMap 提 供 了 出 色 的 put0 以 及 有 反复 时 间 ， 但 get0 
的 性 能 并 不 佳 。 但 是 ， 我 们 为 什么 仍然 需要 使 用 TreeMap 呢 ?这 样 一 
来 ， 我 们 可 以 不 把 它 作 为 Map 使 用 ， 而 作为 创建 顺序 列表 的 一 种 途径 。 








树 的 本 质 在 于 它 总 是 顺序 排列 的 ， 不 必 特 别 进行 排序 ( 它 的 排序 方式 马 

上 就 要 讲 到 〉。 一 旦 十 充 了 一 个 TreeMap， 就 可 以 调用 keySet0 来 获得 键 
的 一 个 Set“ 景 然后 用 toArray0 产 生 包 含 了 那些 键 的 一 个 数组 。 随 
后 ， 可 用 iaa T EANAN. binarySearch() 快 速 查找 排 好 序 的 数组 中 的 内 
容 。 当 然 ， 也 许 只 有 在 HashMap 的 行为 不 可 接受 的 时 候 ， 才 需要 采用 这 
种 做 法 。 因 为 HashMap 的 设计 守则 就 是 进行 快速 的 检索 操作 。 最 后 ， 当 
我 们 使 用 Map 时 ， 首 要 的 选择 应 该 是 HashMap。 只 有 在 极 少 数 情 况 下 才 
需要 考虑 其 他 方法 。 


此 外 ， 在 上 面 那 张 表 里 ， 有 另 一 个 性 能 问题 没有 反映 出 来 。 下 述 程序 用 
于 测试 不 同类 型 Map 的 创建 速度 : 


//: MapCreation.java 


// Demonstrates time differences in Map creation 
package c08.newcollections; 
import java.util.*; 
public class MapCreation { 
public static void main(String[] args) { 
final long REPS = 100000; 
long t1 = System.currentTimeMillis(); 
System.out.print("Hashtable") ; 
for(long i = 0; i < REPS; i++) 
new Hashtable(); 
long t2 = System.currentTimeMillis(); 
System.out.printin(": " + (t2 - t1)); 
t1 = System.currentTimeMillis(); 


System.out.print("TreeMap"); 


for(long i = 0; i < REPS; i++) 

new TreeMap(); 
t2 = System.currentTimeMillis(); 
System.out.println(": " + (t2 - t1)); 
ti = System.currentTimeMillis(); 
System.out.print("HashMap") ; 
for(long i = 0; i < REPS; i++) 

new HashMap(); 
t2 = System.currentTimeMillis(); 
System.out.println(": " + (t2 - t1)); 

As 
} ///:~ 


在 写 这 个 程序 期 间 ，TreeMap 的 创建 速度 比 其 他 两 种 类 型 明显 快 得 多 
(但 你 应 亲自 尝试 一 下 ， 因 为 据说 新 版 本 可 能 会 改善 ArrayMap 的 性 

fE) 。 考 虑 到 这 方面 的 原因 ， 同 时 由 于 前 述 TreeMap 出 色 的 putO 性 能 ， 
所 以 如 果 需 要 创建 大 量 Map， 而 且 只 有 在 以 后 才 需 要 涉及 大 量 检索 操 
作 ， 那 么 最 佳 的 策略 就 是 : 创建 和 填充 TreeMap; 以 后 检索 量 增 大 的 时 
候 ， 再 将 重要 的 TreeMap 转 换 成 HashMap 使 用 HashMap(Map) 构 建 
医 。 同 样 地 ， 只 有 在 事实 证 明确 实 存在 性 能 瓶 饭 后 ， 才 应 关心 这 些 方 面 
的 问题 一 一 先 用 起 来 ， 再 根据 需要 加 快速 度 。 


8.7.6 未 文 持 的 操作 


利用 static( 静 态 〉 数组 Arrays.toList()， 也 许 能 将 一 个 数组 转换 成 List， 
如 下 所 示 : 


//: Unsupported.java 

















// Sometimes methods defined in the Collection 
// interfaces don't work! 
package c08.newcollections; 
import java.util.*; 
public class Unsupported { 
private static String[] s = { 
"one", "two", "three", "four", "five", 
"six", "seven", "eight", "nine", "ten", 
}; 
static List a = Arrays.toList(s); 
static List a2 = Arrays.toList( 
new String[] { s[3], s[4], s[5] }); 
public static void main(String[] args) { 
Collectioni.print(a); // Iteration 
System.out.printin( 
"a.contains(" + s[O] + ") =" + 
a.contains(s[0])); 
System.out.printin( 
"a.containsAll(a2) = " + 
a.containsAll(a2)); 
System.out.println("a.isEmpty() = " + 
a.isEmpty()); 


System.out.printin( 


"a.indexof(" + s[5] +") =" + 
a.indexOf(s[5])); 
// Traverse backwards: 
ListIterator lit = a.listIterator(a.size()); 
while(lit.hasPrevious() ) 
System.out.print(lit.previous()); 
System.out.printin(); 
// Set the elements to different values: 
for(int i = 0; i < a.size(); i++) 
a.set(i, "47"); 
Collection1.print(a); 
// Compiles, but won't run: 
lit.add("X"); // Unsupported operation 
a.clear(); // Unsupported 
a.add("eleven"); // Unsupported 
a.addAll(a2); // Unsupported 
a.retainAll(a2); // Unsupported 
a.remove(s[0]); // Unsupported 
a.removeAll(a2); // Unsupported 


} 
Sdn 


从 中 可 以 看 出 ， 实 际 只 实现 了 Collection 和 List 接 口 的 一 部 分 。 剩 余 的 方 


法 导致 了 不 受 欢 迎 的 一 种 情况 ， 名 为 UnsupportedOperationEXception 。 

在 下 一 章 里 ， 我 们 会 讲述 违例 的 详细 情况 ， 但 在 这 里 有 必要 进行 一 下 简 
单 说 明 。 这 里 的 关键 在 于 “集合 接口 ”， 以 及 新 集合 库 内 的 另 一 些 接口 ， 

它们 都 包含 了 “可 选 的 * 方 法。 在 实现 那些 接口 的 集合 类 中 ， 或 者 提供 、 
或 者 没有 提供 对 那些 方法 的 文 持 。 知 调用 一 个 未 获 文 持 的 方法 ， 就 会 导 
致 一 个 UnsupportedOperationException (操作 未 支持 违例 ) ， 这 表明 出 
现 了 一 个 编程 错误 。 


大 家 或 许 会 觉得 奇怪 ， 不 是 说 “接口 ?和 基础 类 最 大 的 “卖点 ?就 是 它们 许 
诡 这 些 方 法 能 产生 一 些 有 意义 的 行为 吗 ? 上 述 违例 破坏 了 那个 许诺 一 
它 调用 的 一 部 分 方法 不 仅 不 能 产生 有 意义 的 行为 ， 而 且 还 会 中 止 程序 的 
运行 。 在 这 些 情况 下 ， 类 型 的 所 谓 安全 保证 似乎 显得 一 钱 不 值 ! 但 是 ， 
情况 并 没有 想象 的 那么 坏 。 通 过 Collection，List，Set 或 者 Map， 编 译 器 
仍然 限制 我 们 只 能 调用 那个 接口 中 的 方法 ， 所 以 它 和 Smalltalk 还 是 存在 
一 些 区 别 的 (在 Smalltalk 中 ， 可 为 任何 对 象 调 用 任何 方法 ， 而 且 只 有 在 
运行 程序 时 才 知 道 这 些 调用 是 否 可 行 )。 除 此 以 外 ， 以 Collection 作 为 自 
变量 的 大 多 数 方法 只 能 从 那个 集合 中 读 取 数据 Collection 的 所 

有 “read” 方 法 都 不 是 可 选 的 。 


这 样 一 来 ， 系 统 就 可 避免 在 设计 期 间 出 现 接口 的 冲突 。 而 在 集合 库 的 其 
他 设计 方 采 中 ， 最 终 经 常 部 会 得 到 数量 过 多 的 接口 ， 用 它们 描述 基本 方 
案 的 每 一 种 变化 形式 ， 所 以 学 习 和 和 掌握 显得 非常 困难 。 有 些 时 候 ， 甚 至 
难于 捕捉 接口 中 的 所 有 特殊 情况 ， 因 为 人 们 可 能 设计 出 任何 新 接口 。 但 
Java 的 “不 文 持 的 操作 ?方法 却 达到 了 新 集合 库 的 一 个 重要 设计 目标 : 易 
于 学 习 和 使 用 。 但 是 ， 为 了 使 这 一 方法 真正 有 效 ， 却 需 满足 下 述 条 件 : 


(1) UnsupportedOperationException 必 须 属 于 一 种 “非常 ?事件 。 也 就 是 

对 于 大 多 数 类 来 说 ， 所 有 操作 都 应 是 可 行 的 。 只 有 在 一 些 特殊 情况 
、 两 个 操作 才 可 能 未 获 文 持 。 新 集合 库 满 足 了 这 一 条 件 ， 因 为 绝 

hs 数 时 候 用 到 的 类 一 ArrayList，LinkedList，HashList 和 HashMap， 

以 及 其 他 集合 方案 一 一 都 提供 了 对 所 有 操作 的 支持 。 但 是 ， 如 果 想 新 建 

一 个 集合 ， 同 时 不 想 为 集合 接口 中 的 所 有 方法 都 提供 有 意义 的 定义 ， 同 

~ 与 现 有 库 配 合 ， 这 种 设计 方法 也 确实 提供 了 一 个 “后 门 ” 可 以 利 









































(2) 知 一 个 操作 未 获 文 持 ， 那 么 UnsupportedOperationException (RFF 
的 操作 违例 ) 极 有 可 能 在 实现 期 间 出 现 ， 则 不 是 在 产品 已 交付 给 客户 以 
后 才 会 出 现 。 它 毕竟 指出 的 是 一 个 编程 错误 = 











类 。 这 一 点 不 能 十 分 确定 ， 通 过 也 可 以 看 出 这 种 方案 的 “试验 ”特征 一 一 
只 有 经 过 多 次 试验 ， 才 能 找 出 最 理想 的 工作 方式 。 


在 上 面 的 例子 中 ，Arrays.toList(0 产 生 了 一 个 List〈 列 表 ) ， 访 列表 是 由 
一 个 固定 长 度 的 数组 后 推出 来 的 。 因 此 唯一 能 够 文 持 的 就 是 那些 不 改变 
数组 长 度 的 操作 。 在 另 一 方面 ， 知 请 求 一 个 新 接口 表达 不 同 种 类 的 行为 
(可 能 叫 作 “FixedSizeList” 一 一 固定 长 度 列表 〉 ， 就 有 遭遇 更 大 的 复杂 
程度 的 危险 。 这 样 一 来 ， 以 后 试图 使 用 库 的 时 候 ， 很 快 束 会 发 现 自己 不 
知 从 何 处 下 手 。 


对 那些 采用 Collection，List，Set 或 者 Map 作 为 参数 的 方法 ， 它 们 的 文档 


应 当 指 出 哪些 可 选 的 方法 是 必须 实现 的 。 举 个 例子 来 说 ， 排 序 要 求实 现 
set() 和 Iterator.set() 方 法 ， 但 不 包括 add() 和 remove()。 


8.7.7 排序 和 搜索 

Java 1.2 添加 了 自己 的 一 套 实用 工具 ， 可 用 来 对 数组 或 列表 进行 排列 和 
搜索 。 这 些 工 具 都 属于 两 个 新 类 的 “静态 ”方法 。 这 两 个 类 分 别 是 用 于 排 
序 和 搜索 数组 的 Arrays， 以 及 用 于 排序 和 搜索 列表 的 Collections。 

1. 数组 

Arrays 类 为 所 有 基本 数据 类 型 的 数组 提供 了 一 个 过 载 的 sort0 和 
binarySearchO0， 它 们 亦 可 用 于 String 和 Object。 下 面 这 个 例子 显示 出 如 何 


排序 和 搜索 一 个 字 节 数组 〈 其 他 所 有 基本 数据 类 型 都 是 类 似 的 ) 以 及 一 
个 String 数 组 : 


//: Array1.java 








// Testing the sorting & searching in Arrays 
package c08.newcollections; 

import java.util.*; 

public class Array1 { 


static Random r = new Random(); 


static String ssource = 
"ABCDEFGHI JKLMNOPQRSTUVWXYZ" + 
"abcdefghijkimnopqrstuvwxyz"; 
static char[] src = ssource.toCharArray(); 
// Create a random String 
public static String randString(int length) { 
char[] buf = new char[length]; 
int rnd; 
for(int i = 0; i < length; i++) { 
rnd = Math.abs(r.nextInt()) % src.length; 
buf[i] = src[rnd]; 
} 
return new String(buf); 
} 
// Create a random array of Strings: 
public static 
String[] randStrings(int length, int size) { 
String[] s = new String[size]; 
for(int i = 0; i < size; i++) 
s[i] = randString(length); 
return s; 


} 


public static void print(byte[] b) { 


for(int i = 0; i < b.length; i++) 
System.out.print(b[i] + " "); 
System.out.println(); 
} 
public static void print(String[] s) { 
for(int 1 = 0; i < s.length; i++) 
System.out.print(s[i] + " "); 
System.out.println(); 
} 
public static void main(String[] args) { 
byte[] b = new byte[15]; 
r.nextBytes(b); // Fill with random bytes 
print(b); 
Arrays.sort(b); 
print(b); 
int loc = Arrays.binarySearch(b, b[10]); 
System.out.printin("Location of " + b[10] 
"= "+ loc); 
// Test String sort & search: 
String[] s = randStrings(4, 10); 
print(s); 
Arrays.sort(s); 


print(s); 


loc = Arrays.binarySearch(s, s[4]); 
System.out.println("Location of " + s[4] + 
" 二 " + loc); 


} 
} ///:~ 


类 的 第 一 部 分 包含 了 用 于 产生 随机 字 串 对 象 的 实用 工具 ， 可 供 选 择 的 随 
机 字母 保存 在 一 个 字符 数组 中 。randString0 返 回 一 个 任意 长 度 的 字 串 ; 
而 readStringsO 创 建 随机 字 串 的 一 个 数组 ， 同 时 给 定 每 个 字 串 的 长 度 以 
及 希望 的 数组 大 小 。 两 个 print() 方 法 简化 了 对 示范 数组 的 显示 。 在 
main() 中 ，Random.nextBytes() 用 随机 选择 的 字 节 填充 数组 自 变 量 ( 没 有 
对 应 的 Random 方 法 用 于 创建 其 他 基本 数据 类 型 的 数组 ) 。 获 得 一 个 数 
组 后 ， 便 可 发 现 为 了 执行 sort() 或 者 binarySearch()， 只 需 发 出 一 次 方法 调 
用 即 可 。 与 binarySearch() 有 关 的 还 有 一 个 重要 的 警告 ， 车 在 执行 一 次 
a a 便 会 发 生 不 可 预测 的 行为 ， 其 中 甚至 包 
舌 无 限 循环 。 


对 String 的 排序 以 及 搜索 是 相似 的 ， 但 在 运行 程序 的 时 候 ， 我 们 会 注意 
到 一 个 有 趣 的 现象 : 排序 遵守 的 是 字典 顺序 ， 亦 即 大 写字 母 在 字符 集中 
位 于 小 写字 母 的 前 面 。 因 此 ， 上 所 有 大 写字 母 都 位 于 列表 的 最 前 面 ， 后 面 
和 Z 居 然 位 于 a 的 前 面 。 似 乎 连 电话 短 也 是 这 样 排序 


2. 可 比较 与 比较 器 


但 假 知 我们 不 满足 这 一 排序 方式 ， 又 该 如 何 处 理 呢 ? 例如 本 书后 面 的 索 
引 ， 如 果 必 须 对 以 A 或 a 开 头 的 词 条 分 别 到 两 处 地 方 查 看 ， 那 么 肯定 会 使 
读者 颅 不 耐烦 。 


各 想 对 一 个 Object 数组 进行 排序 ， 那 么 必须 解决 一 个 问题 。 根 据 什 么 来 
判定 两 个 Object 的 顺序 呢 ? 不 笠 的 是 ， 最 初 的 Java 设 计 者 并 不 认为 这 是 
一 个 重要 的 问题 ， 人 否则 就 已 经 在 根 类 Object 里 定义 它 了 。 这 样 造 成 的 一 
个 后 果 便 是 : 必须 从 外 部 进行 Object 的 排序 ， 而 且 新 的 集合 库 提 供 了 实 
现 这 一 操作 的 标准 方式 《最 理想 的 是 在 Object 里 定义 它 ) 。 






































针对 Object 数组 〈 以 及 String， 它 当然 属于 Object 的 一 种 ) ， 可 使 用 一 个 
sort()， 并 令 其 接纳 男 一 个 参数 : 实现 了 Comparator 接 口 ( 即 “比较 器 ” 接 
口 ， 新 集合 库 的 一 部 分 ) 的 一 个 对 象 ， 并 用 它 的 单个 compare() 方 法 进行 
Lasts Mt Sle seat a, 一 一 各 第 一 
个 参数 小 于 第 二 个 ， 返 回 一 个 负 整 数 ; AHS, 零 ; 各 第 一 个 参数 
大 于 第 二 个 ， 则 返回 正 整数 。 基 于 这 一 规则 ， 
可 重新 写 过 ， SEER LER MME, 


//: AlphaComp.java 


// Using Comparator to perform an alphabetic sort 
package c08.newcollections; 
import java.util.*; 
public class AlphaComp implements Comparator { 
public int compare(Object o1, Object 02) { 
// Assume it's used only for Strings... 
String si = ((String)o1).toLowerCase()j; 
String s2 = ((String)o2).toLowerCase()j; 
return si.compareTo(s2); 
} 
public static void main(String[] args) { 
String[] s = Arrayi1.randStrings(4, 10); 
Array1.print(s); 
AlphaComp ac = new AlphaComp(); 
Arrays.sort(s, ac); 


Array1.print(s); 


// Must use the Comparator to search, also: 
int loc = Arrays.binarySearch(s, s[3], ac); 
System.out.println("Location of " + s[3] + 
"=" 4 loc); 


} 
} ///:~ 


通过 造型 为 String，compare() 方 法 会 进行 “暗示 ”性 的 测试 ， 保 证 自己 操 
作 的 只 能 是 String 对 象 一 一 运行 期 系统 会 捕获 任何 差错 。 将 两 个 字 串 都 
强迫 换 成 小 写 形式 后 ，String.compareTo() 方 法 会 产生 预期 的 结 


若 用 自己 的 Comparator 来 进行 一 次 sort0， 那 么 在 使 用 binarySearchO 时 必 
须 使 用 那个 相同 的 Comparator。 


Arrays 类 提供 了 男 一 个 sort() 方 法 ， 它 会 采用 单个 自 变 量 : 一 个 Object 数 
组 ， 但 没有 Comparator。 这 个 sort0 方 法 也 必须 用 同样 的 方式 来 比较 两 个 
Object。 通 过 实现 Comparable 接 口 ， 它 采用 了 赋予 一 个 类 的 “ 目 然 比较 方 
法 ”。 这 个 接口 含有 单独 一 个 方法 compareTo()， 能 分 别 根据 它 小 
于 、 等 于 或 者 大 于 目 变 量 而 返回 负数 、 零 或 者 正 数 ， 从 而 实现 对 象 的 比 
较 。 下 面 这 个 例子 简单 地 阐 示 了 这 一 点 : 


//: CompClass.java 




















// A class that implements Comparable 

package c08.newcollections; 

import java.util.*; 

public class CompClass implements Comparable { 
private int 1; 


public CompClass(int ii) { i = ii; } 


public int compareTo(Object o) { 
// Implicitly tests for correct type: 
int argi = ((CompClass)o).1; 
if(i == argi) return 0; 
if(i < argi) return -1; 
return 1; 
} 
public static void print(Object[] a) { 
for(int i = 0; i < a.length; i++) 
System.out.print(a[i] + " "); 
System.out.println(); 
} 
public String toString() { return i + ""; } 
public static void main(String[] args) { 
CompClass[] a = new CompClass[20]; 
for(int i = 0; i < a.length; i++) 
a[i] = new CompClass( 
(int)(Math.random() *100)); 
print(a); 
Arrays.sort(a); 
print(a); 
int loc = Arrays.binarySearch(a, a[3]); 


System.out.println("Location of " + a[3] 


" = " + loc); 


Í 
} ///:~ 





当然 ， 我 们 的 compareTo() 方 法 亦 可 根据 实际 情况 增 大 复杂 程度 。 
3. 列表 


可 用 与 数组 相同 的 形式 排序 和 搜索 一 个 列表 〈List) 。 用 于 排序 和 搜索 
列表 的 静态 方法 包含 在 类 Collections 中 ， 但 它们 拥有 与 Arrays 中 差不多 
的 签名 : sort(List) 用 于 对 一 个 实现 了 Comparable 的 对 象 列表 进行 排序 ; 
binarySearch(List,Object) 用 于 查找 列表 中 的 某 个 对 象 ; 
sort(List,Comparator) 利 用 一 个 “比较 器 ”对 一 个 列表 进行 排序 ， 而 
binarySearch(List,Object,Comparator) 则 用 于 查找 那个 列表 中 的 一 个 对 象 

(注释 @) 。 下 面 这 个 例子 利用 了 预先 定义 好 的 CompClass 和 
AlphaComp 来 示范 Collections 中 的 各 种 排序 工具 : 


//: ListSort.java 

















// Sorting and searching Lists with 'Collections' 
package c08.newcollections; 
import java.util.*; 
public class ListSort { 
public static void main(String[] args) { 
final int SZ = 20; 
// Using "natural comparison method": 
List a = new ArrayList(); 
for(int i = 0; i < SZ; i++) 


a.add(new CompClass( 


(int)(Math.random() *100))); 

Collectioni.print(a); 
Collections.sort(a); 
Collectioni.print(a); 
Object find = a.get(SZ/2); 
int loc = Collections.binarySearch(a, find); 
System.out.println("Location of " + find + 

Hh see SOG ya 
// Using a Comparator: 
List b = new ArrayList(); 
for(int i = 0; i < SZ; i++) 

b.add(Array1.randString(4)); 
Collectioni.print(b); 
AlphaComp ac = new AlphaComp(); 
Collections.sort(b, ac); 
Collectioni.print(b); 
find = b.get(SZ/2); 
// Must use the Comparator to search, also: 
loc = Collections.binarySearch(b, find, ac); 
System.out.println("Location of " + find + 


" 二 " + loc) $ 


} ///:~ 


©: 在 本 书写 作 时 ， 己 宣布 了 一 个 新 的 Collections.stableSort()， 可 用 它 
进行 合并 式 排 序 ， 但 还 没有 它 的 测试 版 问世 。 


这 些 方法 的 用 法 与 在 Arrays 中 的 用 法 是 完全 一 致 的 ， 只 是 用 一 个 列表 代 
B T BUA 


TreeMap 也 必须 根据 Comparable 或 者 Comparator 对 上 自己 的 对 象 进 行 排 
序 ; 





8.7.8 实用 工具 


Collections 类 中 含有 其 他 大 量 有 用 的 实用 工具 : 


enumeration(Collection)||Produces an old-style Enumeration for the 
argument. 


max(Collection) Produces the maximum or minimum element in 
the argument using the natural comparison 
method of the objects in the Collection. 


max(Collection, Produces the maximum or minimum element in 
Comparator) the Collection using the Comparator. 


argument List that is a window into that 
argument with indexes starting at min and 
stopping just before max. 





enumeration(Collection) 为 和 目 变量 产生 原始 风格 的 Enumeration 〈 枚 举 ) 


max(Collection), min(Collection) 在 自 变量 中 用 集合 内 对 象 的 自然 比较 方 
法 产生 最 大 或 最 小 元 素 


max(Collection,Comparator), min(Collection,Comparator) ”在 集合 内 用 比 
较 占 产生 最 大 或 最 小 元 素 


oo n, Object 0) 返回 长 度 为 n 的 一 个 不 可 变 列表 ， 它 的 所 有 句柄 
alajo 


subList(List,int min,int max) 返回 由 指定 参数 列表 后 推 得 到 的 一 个 新 列 
表 。 可 将 这 个 列表 想象 成 一 个 “窗口 ?， 它 自 索 引 为 min 的 地 方 开始 ， 正 
好 结束 于 max 的 前 面 


注意 min) Fmax) Ai 随同 Collection 对 象 工作 的 ， 而 非 随同 List， 所 以 
不 必 担 心 Collection 是 人 否 需 要 排序 〈 就 象 早先 指出 的 那样 ， 在 执行 一 次 

binarySearch() BU = = El 搜索 之 前 ， 必 须 对 一 个 List 或 者 一 个 数 
组 执行 sort())。 


1. 使 Collection 或 Map 不 可 修改 


s BIE Collectionmt Maph] = 1A 读 ” 版 本 显得 更 有 利 一 些 。 
ee 类 允许 我 们 达到 这 个 目标 ， 方 法 是 将 原 容器 传递 进入 一 个 
方法 ， 并 令 其 传 回 一 个 只 读 版 本 。 这 个 方法 共有 四 种 变化 形式 ， 分 别 用 
于 Collection (如 果 不 想 把 集合 当 作 一 种 更 特殊 的 类 型 对 待 )、List、Set 
以 及 Map。 下 面 这 个 例子 演示 了 为 它们 分 别 构建 只 读 版 本 的 正确 方法 : 


//: ReadOnly.java 























// Using the Collections.unmodifiable methods 
package c08.newcollections; 

import java.util.*; 

public class ReadOnly { 


public static void main(String[] args) { 


Collection c = new ArrayList(); 
Collectioni.fill(c); // Insert useful data 
c = Collections.unmodifiableCollection(c); 
Collectioni.print(c); // Reading is OK 


//\ c.add("one"); // Can't change it 


List a = new ArrayList(); 

Collection1.fill(a); 

a = Collections.unmodifiableList(a); 

ListIterator lit = a.listIterator(); 

System.out.printin(lit.next()); // Reading OK 
//! lit.add("one"); // Can't change it 

Set s = new HashSet(); 

Collection1.fill(s); 

s = Collections.unmodifiableSet(s); 

Collectioni.print(s); // Reading OK 


//! s.,add("one"); // Can't change it 


Map m = new HashMap(); 

Map1.fill(m, Map1.testDatat) ; 

m = Collections.unmodifiableMap(m); 
Mapi.print(m); // Reading OK 


//! m.put("Ralph", "Howdy!"); 


} 
于 LA 


对 于 每 种 情况 ， 在 将 其 正式 变 为 只 读 以 前 ， 都 必须 用 有 有 效 的 数据 填充 
容 帮 。 一 旦 载 入 成 功 ， 最 佳 的 做 法 就 是 用 “不 可 修改 ”调用 产生 的 句柄 葵 
换 现 有 的 句柄 。 这 样 做 可 有 效 避 人 免 将 其 变 成 不 可 修改 后 不 慎 改 变 其 中 的 
内 容 。 在 为 一 方面 ， 该 工具 也 允许 我 们 在 一 个 类 中 将 能 够 修改 的 容 句 保 
持 为 private 状 态 ， 并 可 从 一 个 方法 调用 中 返回 指向 那个 容器 的 一 个 只 读 
句柄 。 这 样 一 来 ， 虽 然 我 们 可 在 类 里 修改 它 ， 但 其 他 任何 人 都 只 能 读 。 


为 特定 类 型 调用 “不 可 修改 ”的 方法 不 会 造成 编译 期 间 的 检查 ， 但 一 旦 发 
生 任 何 变化 ， 对 修改 特定 容器 的 方法 的 调用 便 会 产生 一 个 
UnsupportedOperationException 违 例 。 

















2. Collection 或 Map 的 同步 


synchronized 关 键 字 是 “多 线程 ”机 制 一 个 非常 重要 的 部 分 。 我 们 到 第 14 
章 才 会 对 这 一 机 制作 深入 的 探讨 。 在 这 儿 ， 大 家 只 需 注意 到 Collections 
类 提供 了 对 整个 容器 进行 自动 同步 的 一 种 途径 。 它 的 语法 与 “不 可 修 
改 ” 的 方法 是 类 似 的 : 


//: Synchronization,java 





// Using the Collections.synchronized methods 
package c08.newcollections; 
import java.util.*; 
public class Synchronization { 
public static void main(String[] args) { 
Collection c = 
Collections.synchronizedCollection( 


new ArrayList()); 


List list = Collections.synchronizedList( 
new ArrayList()); 

Set s = Collections.synchronizedSet ( 
new HashSet()); 

Map m = Collections.synchronizedMap ( 
new HashMap()); 


} 
SA 


在 这 种 情况 下 ， 我 们 通过 适当 的 “同步 ”方法 直接 传递 新 容器 ;这 样 做 可 
避免 不 愤 和 暴露 出 未 同步 的 版 本 


新 集合 也 提供 了 能 防止 多 个 进程 同时 修改 一 个 容器 内 容 的 机 制 。 辱 在 一 
个 容器 里 有 反复， 同时 男 一 些 进程 介入 ， 并 在 那个 容器 中 插入 、 删 除 或 修 
改 一 个 对 象 ， 便 会 面临 发 生 冲突 的 危险 。 我 们 可 能 已 传递 了 那个 对 象 ， 
可 能 它 位 位 于 我 们 前 面 ， 可 能 容 右 的 大 小 在 我 们 调用 size() 后 已 发 生 了 

缩 了 面临 下 险 。 针 对 这 个 问题 ， 新 的 集合 库 集 
成 了 一 套 解 决 的 机 制 ， 能 查 出 除 我 们 的 进程 自己 需要 负责 的 之 外 的 、 对 
容器 的 其 他 任何 修改 。 若 探测 到 有 其 他 方面 也 准备 修改 容器 便 会 并 即 
产生 一 个 ConcurrentModificationException (并 发 修改 违例 ) 。 我 们 将 这 
一 机 制 称 为 “立即 失败 ” 它 并 不 用 更 复杂 的 算法 在 “以 后 :个 测 问题 ， 
而 是 “并 即 ” 产 生 违例 。 

















8.8 Ih Zi 


下 面 复 习 一 下 由 标准 Java〈1.0 和 1.1) 库 提 供 的 集合 〈BitSet 未 包括 在 这 
里 ， 因 为 它 更 象 一 种 负 有 特殊 使 命 的 类 ) : 


(1) 数组 包含 了 对 象 的 数字 化 索引 。 它 容纳 的 是 一 种 已 知 类 型 的 对 象 ， 
所 以 在 查找 一 个 对 象 时 ， 不 必 对 结果 进行 造型 处 理 。 数 组 可 以 是 多 维 
ay De 但 是 ， 一 旦 把 它 创建 好 以 后 ， 大 小 便 
` 能 变化 了 。 


(2) Vector (KE) 也 包含 了 对 象 的 数字 索引 可 将 数组 和 Vector 想象 

成 随机 访问 集合 。 当 我 们 加 入 更 多 的 元 素 时 ，Vector 能 够 目 动 改变 上 自身 

的 大 小 。 但 Vector 只 能 容纳 对 象 的 句柄 ， 所 以 它 不 可 包含 基本 数据 类 

和 必须 对 结果 进行 造型 
H, 


(3) Hashtable (HIX) 属于 Dictionary (字典 ) 的 一 种 类 型 ， 是 一 种 将 
对 象 “〈 而 不 是 数字 ) 同 其 他 对 象 关 联 到 一 起 的 方式 。 散 列表 也 文 持 对 对 
象 的 随机 访问 ， 事 实 上 ， 它 的 整个 设计 方案 都 在 突出 访问 的 “高 速度 ”。 


(4) Stack CEFR) 是 一 种 “后 入 先 出 ”(LIFO》 的 队列 。 


在 你 曾经 熟悉 数据 结构 ， 可 能 会 疑惑 为 何 没 看 到 一 套 更 大 的 集合 。 从 功 
能 的 角度 出 用， 你 真 的 需要 一 套 更 大 的 集合 吗 ? 对 于 Hashtable， 可 将 任 
何 东 西 置 入 其 中 ， 并 以 非常 快 的 速度 检索 ， 对 于 Enumeration 〈 枚 举 ) ， 
可 遍历 一 个 序列 ， 并 对 其 中 的 每 个 元 素 都 采取 一 个 特定 的 操作 。 那 是 一 
种 功能 足够 强劲 的 工具 。 


但 Hashtable 没 有 “顺序 ”的 概念 。Vector 和 数组 为 我 们 提供 了 一 种 线性 顺 
序 ， 但 知 要 把 一 个 元 素 揪 入 它们 任何 一 个 的 中 部 ， 一 般 都 要 付出 “ 惨 
重 ” 的 代价 。 除 此 以 外 ， 队 列 、 拆 散 队列 、 优 先 级 队列 以 及 树 都 涉及 到 
元 素 的 “排序 一 一 并 非 仅 仅 将 它们 置 入 ， 以 便 以 后 能 按 线性 顺序 查找 或 
移动 它们 。 这 些 数据 结构 也 非常 有 用 ， 这 也 正 是 标准 C++ 中 包含 了 它们 
的 原因 。 考 虑 到 这 个 原因 ， 只 应 将 标准 Java 库 的 集合 看 作 目 己 的 一 个 起 
点 。 而 且 倘 车 必须 使 用 Java 1.0 或 1.1， 则 可 在 需要 超越 它们 的 时 候 使 用 
JGL. 


























如 果 能 使 用 Java 1.2， 那么 只 使 用 新 集合 即 可 ， 它 一 般 能 满足 我 们 的 所 

需要 。 注 意 本 书 在 Java 1.1 身 上 人 花 了 大 量 篇 幅 ， 所 以 书 中 用 到 的 大 量 
集合 都 是 只 能 在 Javal.1 中 用 到 的 那些 : Vector#llHashtable. it H AIK 
看 ， 这 是 一 个 不 得 以 而 为 之 的 做 法 。 但 是 ， 这 样 处 理 亦 可 提供 与 老 Java 
代码 更 出 色 的 向 后 兼容 能 力 。 知 要 用 Javal1.2 写 新 代码 ， 新 的 集合 往往 能 
更 好 地 为 你 服务 。 


8.9 练习 


(1) 新 建 一 个 名 为 Gerbil 的 类 ， 在 构建 器 中 初始 化 一 个 int 
gerbilNumber 类似 本 章 的 Mouse 例 子 )。 为 其 写 一 个 名 为 hop() 的 方 
法 ， 用 它 打 印 出 符合 hopO 条 件 的 Gerbijl 的 编号 。 建 一 个 Vector， 并 为 
Vector 添 加 一 系列 Gerbil 对 象 。 现 在 ， 用 elementAt() 方 法 在 Vector 中 遍 
历 ， 并 为 每 个 Gerbil 都 调用 hop()。 


(2) 修改 练习 1， 用 Enumeration 在 调用 hopO 的 同时 遍历 Vector。 


(3) 在 AssocArray.java 中 ， 修 改 这 个 例子 ， 令 其 使 用 一 个 Hashtable， 而 不 
是 AssocArray。 


(4) ”获取 练习 1 用 到 的 Gerbil 类 ， 改 为 把 它 置 入 一 个 Hashtable， 然 后 将 
Gerbil 的 名 称 作 为 一 个 String〈 键 ) 与 置 入 表格 的 每 个 Gerbi〈 值 ) 都 关 
联 起 来 。 获 得 用 于 keys0 的 一 个 Enumeration， 并 用 它 在 Hashtable 里 遍 

历 ， 查 找 每 个 键 的 Gerbil， 打 印 出 键 ， 然 后 将 gerbil 告 诉 给 hop()。 


(5) “修改 第 7 章 的 练习 1， 用 一 个 Vector 容纳 Rodent〈 吐 齿 动 物 ) ， 并 用 
Enumeration 在 Rodent 序 列 中 裔 历 。 记 住 Yector 只 能 容纳 对 象 ， 所 以 在 访 
问 单独 的 Rodent 时 必须 采用 一 个 造型 (如 RTTI) 。 


(6) ” 转 到 第 7 章 的 中 间 位 置 ， 找 到 那个 GreenhouseControls.java (温室 控 
制 ) 例子 ， 该 例 应 该 由 三 个 文件 构成 。 在 Controller.java 中 ， 类 EventSet 
仅 是 一 个 集合 。 修 改 它 的 代码 ， 用 一 个 Stack 代 蔡 EventSet。 当 然 ， 这 时 
可 能 并 不 仅仅 用 Stack 取 代 EventSet 这 样 简 单 ， 也 需要 用 一 个 Enumeration 
过 历 事件 集 。 可 考虑 在 某 些 时 候 将 集合 当 作 Stack 对 待 ， 另 一 些 时 候 则 
当 作 Vector 对 待 这 样 或 许 能 使 事情 变 得 更 加 简单 。 


(7) (有 一 定 挑战 性 ) 在 与 所 有 Java 发 行 包 配套 提供 的 Java 源 码 库 中 找 出 
用 于 Vector 的 源码 。 复 制 这 些 代 码 ， 制 作 名 为 intVector 的 一 个 特殊 版 
本 ， 只 在 其 中 包含 int 数 据 。 思 考 是 否 能 为 所 有 基本 数据 类 型 都 制作 
Vector 的 一 个 特殊 版 本 。 接 下 来 ， 考 虑 假如 制作 一 个 链接 列表 类 ， 令 其 
能 随同 所 有 基本 数据 类 型 使 用 ， 那 么 会 发 生 什 么 情况 。 知 在 Java 中 提供 
了 参数 化 类 型 ， 利 用 它们 便 可 自动 完成 这 一 工作 (还 有 其 他 许多 好 

处 ) 。 




















PIR 违例 天 错 控制 
Java 的 基本 原理 就 是 “形式 错误 的 代码 不 会 运行 ”。 


与 C++ 类 似 ， 捕 获 错误 最 理想 的 是 在 编译 期 间 ， 最 好 在 试图 运行 程序 以 
前 。 然 而 ， 并 非 所 有 错误 都 能 在 编译 期 间 侦 测 到 。 有 些 问题 必须 在 运行 
期 间 解决 ， 让 错误 的 缔结 者 通过 一 些 手续 癌 接 收 者 传递 一 些 适 当 的 信 

恩 ， 使 其 知道 该 如 何 正 确 地 处 理 遇 到 的 问题 。 


在 C++ 和 其 他 早期 语言 中 ， 可 通过 几 种 手续 来 达到 这 个 目的 。 而 且 它们 
通常 是 作为 一 种 规定 建立 起 来 的 ， 而 非 作 为 程序 设计 语言 的 一 部 分 。 典 
型 地 ， 我 们 需要 返回 一 个 值 或 设置 一 个 标志 《位 ) ， 接 收 者 会 检查 这 些 
faking, FUT AAR STAs. AM, RAIN, APR 
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样 想 :“ 是 的 ， 错 误 可 能 会 在 其 他 人 的 代码 中 出 现 ， 但 不 会 在 我 的 代码 
中 ”。 这 样 的 后 果 便 是 他 们 一 般 不 检查 是 否 出 现 了 错误 (有 时 出 错 条 件 
确实 显得 太 愚蠢， 不 值得 检验 ; 注释 四 ) 。 男 一 方面 ， 帮 每 次 调用 一 个 
方法 时 都 进行 全 面 、 细 致 的 错误 检查 ， 那 么 代码 的 可 读 性 也 可 能 大 幅度 
降低 。 由 于 程序 员 可 能 仍然 在 用 这 些 语言 维护 自己 的 系统 ， 所 以 他 们 应 
该 对 此 有 看 深刻 的 体会 :大 按 这 种 方式 控制 错误 ， 那 么 在 创建 大 型 、 健 
壮 、 易 于 维护 的 程序 时 ， 肯 定 会 过 到 不 小 的 阻挠。 


©: C 程 序 员 研究 一 下 printfO 的 返回 值 便 知 端详 。 


解决 的 方法 是 在 错误 控制 中 排除 所 有 偶然 性 ， 强 制 格式 的 正确 。 这 种 方 
法 实际 已 有 很 长 的 历史 ， 因 为 早 在 60 年 代 便 在 操作 系统 里 采用 了 “违例 
控制 手段， 甚至 可 以 妃 湖 到 BASIC 语 言 的 on error goto 语句。 但 C++ 的 
违例 控制 建立 在 Ada 的 基础 上 ， 而 Java 又 主要 建立 在 C++ 的 基础 上 《【〈 尽 
管 它 看 起 来 更 象 Object Pascal) 。 


“违例 ”(Exception〉 这 个 词 表达 的 是 一 种 “例外 ”情况 ， 亦 即 正 常情 况 之 
外 的 一 种 “异常 *。 在 问题 及 生 的 时 候 ， 我 们 可 能 不 知 具体 该 如 何 解 决 ， 
但 肯定 知道 已 不 能 不 顾 一 切 地 继续 下 去 。 此 时 ， 必 须 坚 决 地 停 下 来 ， 并 
由 某 人 、 某 地 指出 发 生 了 什么 事情 ， 以 及 该 采取 何 种 对 策 。 但 为 了 真正 
解决 问题 ， 当 地 可 能 并 没有 足够 多 的 信息 。 因 此 ， 我 们 需要 将 其 移交 给 
更 级 的 负责 人 ， 令 其 作出 正确 的 决定 (类 似 一 个 命令 链 ) 。 















































违例 机 制 的 男 一 项 好 处 就 是 能 够 简化 错误 控制 代码 。 我 们 再 也 不 用 检查 

一 个 特定 的 错误 ， 然 后 在 程序 的 多 处 地 方 对 其 进行 控制 。 此 外 ， 也 不 需 

要 在 方法 调用 的 时 候 检 查 错误 (因为 保证 有 人 能 捕获 这 里 的 错误 ) 。 我 

们 只 需要 在 一 个 地 方 处 理 问题 “违例 控制 模块 ”或 者 “违例 控制 器 *?。 这 

样 可 有 效 减 少 代码 量 ， 并 将 那些 用 于 描述 具体 操作 的 代码 与 专门 纠正 错 
BAKE. 
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制 ， 便 可 正确 使 用 本 书 编写 的 大 量 例 子 。 本 章 向 大 家 介绍 了 用 于 正确 控 
人 
违例。 


9.1 基本 违例 


“违例 条 件 ” 表 示 在 出 现 什 么 问题 的 时 候 应 中 止 方法 或 作用 域 的 继续 。 为 
了 将 违例 条 件 与 普通 问题 区 分 开 ， 违 例 条 件 是 非常 重要 的 一 个 因素 。 在 
普通 问题 的 情况 下 ， 我 们 在 当地 已 拥有 足够 的 信息 ， 可 在 东 种 程度 上 解 
决 磁 到 的 问题 。 而 在 违例 条 件 的 情况 下 ， 却 无 法 继续 下 去 ， 因 为 当地 没 
有 提供 解决 问题 所 需 的 足够 多 的 信息 。 此 时 ， 我 们 能 做 的 唯一 事情 就 是 
跳出 当地 环境 ， 将 那个 问题 委托 给 一 个 更 高 级 的 负责 人 。 这 便 是 出 现 违 
例 时 出 现 的 情况 。 


一 个 简 蛙 的 例子 是 “除法 ”。 如 可 能 被 零 除 ， 残 有 必要 进行 检查 ， 确 保 程 
序 不 会 冒进 ， 并 在 那 种 情况 下 执行 除法 。 但 具体 通过 什么 知道 分 母 是 零 
Ne? 在 那个 特定 的 方法 里 ， 在 我 们 试图 解决 的 那个 问题 的 环境 中 ， 我 们 
或 许 知 道 该 如 何 对 竺 一 个 零 分母 。 但 假如 它 是 一 个 没有 预料 到 的 值 ， 惑 
人 
Ps 


产生 一 个 违例 时 ， 会 发 生 几 件 事情 。 首 先 ， 按 照 与 创建 Java 对 象 一 样 的 
方法 创建 违例 对 象 : 在 内 存 “ 扒 ?里 ， 使 用 new 来 创建 。 随 后 ， 俘 止 当前 
执行 路 径 〈 记 住 不 可 沿 这 条 路 径 继 续 下 去 ) ， 然 后 从 当前 的 环境 中 释放 
出 违例 对 象 的 句柄 。 此 时 ， 违 例 控制 机 制 会 接管 一 切 ， 并 开始 碍 找 一 个 
恰当 的 地 方 ， 用 于 继续 程序 的 执行 。 这 个 恰当 的 地 方便 是 “违例 控制 
器 ”， 它 的 职责 是 从 问题 中 恢复 ， 使 程序 要 么 答 试 妨 一 条 执行 路 径 ， 要 
么 简单 地 继续 。 


作为 产生 违例 的 一 个 简单 示例 ， 大 家 可 思考 一 个 名 为 { 的 对 象 句 柄 。 有 

些 时 候 ， 程 序 可 能 传递 一 个 尚未 初始 化 的 句柄 。 所 以 在 用 那个 对 象 句柄 
调用 一 个 方法 之 前 ， 最 好 进行 一 番 检 查 。 可 将 与 错误 有 关 的 信息 发 送 到 
一 个 更 大 的 场景 中 ， 方 法 是 创建 一 个 特殊 的 对 象 ， 用 它 代表 我 们 的 信 

Eh, FRAY (Throw) 出 我 们 当前 的 场景 之 外 。 这 就 叫 作 “产生 一 个 
违例 ?或 者 “ 掷 出 一 个 违例 ”。 下 面 是 它 的 大 概 形式 : 


if(t == null) 





























throw new NullPointerException(); 


这 样 便 “ 掷 "出 了 一 个 违例 。 在 当前 场景 中 ， 筷 使 我 们 能 放弃 进一步 解决 
该 问题 的 企图 。 该 问题 会 被 转移 到 其 他 更 恰当 的 地 方 解 决 。 准 确 地 说 ， 
那个 地 方 不 久 就 会 显露 出 来 。 


9.1.1 违例 自 变量 


和 Java 的 其 他 任何 对 象 一 样 ， 需 要 用 new 在 内 存 推 里 创建 违例 ， 并 需 调 
用 一 个 构建 问 。 在 所 有 标准 违例 中 ， 存 在 着 两 个 构建 器 : 第 一 个 是 默认 
TE ae 第 二 个 则 需 使 用 一 个 字 串 上 自 变 量 ， 使 我 们 能 在 违例 里 置 入 相关 


if(t == null) 
throw new NullPointerException("t = null"); 
稍 后 ， 字 串 可 用 各 种 方法 提取 出 来 ， 就 象 稍 后 会 展示 的 那样 。 


在 这 儿 ， 关 键 字 throw 会 象 变 戏 法 一 样 做 出 一 系列 不 可 思议 的 事情 。 它 
首先 执行 hew 表 达 式 ， 创 建 一 个 不 在 程序 常规 执行 范围 之 内 的 对 象 。 而 
且 理 所 当然 ， 会 为 那个 对 象 调用 构建 器 。 随 后 ， 对 象 实际 会 从 方法 中 返 
回 一 一 尽管 对 象 的 类 型 通常 并 不 是 方法 设计 为 返回 的 类 型 。 为 深入 理解 
违例 控制 ， 可 将 其 想象 成 男 一 种 返回 机 制 一 一 但 古 不 要 在 这 个 问题 上 深 
守 ， 人 否则 会 遇 到 麻烦 。 通 过 “ 掷 ?出 一 个 违例 ， 亦 可 从 原来 的 作用 域 中 退 
出 。 但 是 会 先 返 回 一 个 值 ， 再 退出 方法 或 作用 域 。 


但 是 ， 与 普通 方法 返回 的 相似 性 到 此 便 全 部 结束 了 ， 因 为 我 们 返回 的 地 
方 与 从 普通 方法 调用 中 返回 的 地 方 是 包 然 有 异 的 (我 们 结束 于 一 个 恰当 
的 违例 控制 器 ， 它 距离 违例 “ 掷 ? 出 的 地 方 可 能 相当 遥远 一 一 在 调用 堆栈 
中 要 低 上 许多 级 ) 。 


此 外 ， 我 们 可 根据 需要 掷 出 任何 类 型 的 “可 掷 ? 对 象 。 典 型 情况 下 ， 我 们 
要 为 每 种 不 同类 型 的 错误 “ 掷 " 出 一 类 不 同 的 违例 。 我 们 的 思路 是 在 违例 
对 象 以 及 挑选 的 违例 对 象 类 型 中 保存 信息 ， 所 以 在 更 大 场景 中 的 茶 个 人 
可 知道 如 何 对 待 我们 的 违例 通常， 唯一 的 信息 古 违例 对 象 的 类 型 ， 而 
违例 对 象 中 保存 的 没什么 意义 ) 。 























9.2 违例 的 捕获 


各 东 个 方法 产生 一 个 违例 ， 必 须 保证 该 违例 能 被 捕获 ， 并 获得 正确 对 
待 。 对 于 Java 的 违例 控制 机 制 ， 它 的 一 个 好 处 束 是 允许 我 们 在 一 个 地 方 
ys Gener ee ceeiceryncraned 
部 的 错误 。 


为 理解 违例 是 如 何 捕获 的 ， 首 先 必须 掌握 “警戒 区 ”的 概念 。 它 代表 一 个 
特殊 的 代码 区 域 ， 有 可 能 产生 违例 ， 并 在 后 面 跟随 用 于 控制 那些 违例 的 





9.2.1 try 块 


看 位 于 一 个 方法 内 部 ， 并 * 扫 ?出 一 个 违例 《或 在 这 个 方法 内 部 调用 的 另 
一 个 方法 产生 了 违例 ) ， 那 个 方法 就 会 在 违例 产生 过 程 中 退出 。 知 不想 
一 个 throw 离 开 方法 ， 可 在 那个 方法 内 部 设置 一 个 特殊 的 代码 块 ， 用 它 
捕获 违例 。 这 就 叫 作 “try 块 "， 因 为 要 在 这 个 地 方 “尝试 ”各 种 方法 调用 。 
try 块 属于 一 种 普通 的 作用 域 ， 用 一 个 try 关 键 字 开头 : 

try { 

// 可 能 产生 违例 的 代码 

} 

若 用 一 种 不 支持 违例 控制 的 编程 语言 全 面 检 查 错误 ， 必 须 用 设置 和 错误 
检测 代码 将 每 个 方法 都 包围 起 来 一 一 即便 多 次 调用 相同 的 方法 。 而 在 使 
用 了 违例 控制 技术 后 ， 可 将 所 有 东西 都 置 入 一 个 try 块 内 ， 在 同一 地 点 捕 


获 所 有 违例 。 这 样 便 可 极 大 简化 我 们 的 代码 ， 并 使 其 更 易 辨 恋 ， 因 为 代 
码 本 身 要 达到 的 目标 再 也 不 会 与 党 复 的 错误 检查 混 消 。 


9.2.2 违例 控制 器 


当然 ， 生 成 的 违例 必须 在 某 个 地 方 中 止 。 这 个 “地 方 ” 便 是 违例 控制 颖 或 
者 违例 控制 模块 。 而 且 针 对 想 捕 获 的 每 种 违例 类 型 ， 都 必须 有 一 个 相应 
的 违例 控制 器 。 违 例 控制 器 紧 接 在 try 块 后 面 ， 且 用 catch (43k) 关键 
字 标 记 。 如 下 所 示 : 








try { 


// Code that might generate exceptions 


WY 


catch(Typei id1) { 


// Handle exceptions of Typet 


WY 


catch(Type2 id2) { 


// Handle exceptions of Type2 


WY 


catch(Type3 id3) { 


// Handle exceptions of Type3 








} 
// etc... 
每 个 catch 从 句 即 违例 控制 器 一 一 都 类 似 一 个 小 型 方法 ， 它 需要 采用 


一 个 (而 且 只 有 一 个 ) 特定 类 型 的 自 变 量 。 可 在 控制 器 内 部 使 用 标识 符 
(id1，id2 等 等 ) ， 吏 象 一 个 普通 的 方法 上 自 变量 那样 。 我 们 有 时 也 根本 
不 使 用 标识 符 ， 因 为 违例 类 型 已 提供 了 足够 的 信息 ， 可 有 效 处 理 违 例 。 
但 即使 不 用 ， 标 识 符 也 必须 就 位 。 


控制 絮 必 须 “ 紧 接 ” 在 try 块 后 面 。 夺 “ 撕 ” 出 一 个 违例 ， 违 例 控制 机 制 就 会 
搜寻 目 变 量 与 违例 类 型 相符 的 第 一 个 控制 器 。 随 后 ， 它 会 进入 那个 catch 
从 句 ， 并 认为 违例 已 得 到 控制 (一 旦 catch 从 句 结 束 ， 对 控制 器 的 搜索 也 
会 停止 ) 。 只 有 相符 的 catch 从 句 才 会 得 到 执行 ， 它 与 switch 语 句 不 同 ， 

后 者 在 每 个 case 后 都 需要 一 个 break 命 令 ， 防 止 误 执行 其 他 语句 。 


在 try 块 内 部 ， 请 注意 大 量 不 同 的 方法 调用 可 能 生成 相同 的 违例 ， 但 只 需 


要 一 个 控制 右 。 
1. 中 断 与 恢复 


在 违例 控制 理论 中 ， 共 存在 两 种 基本 方法 。 在 “中 断 ” 方 法 中 (Java 和 
C++ 提供 了 对 这 种 方法 的 文 持 ) ， 我 们 假定 错误 非常 关键 ， 没 有 办 法 返 




















回 违例 发 生 的 地 方 。 无 论 谁 只 要 “ 掷 ?" 出 一 个 违例 ， 就 表明 没有 办 法 补救 
错误 ， 而 且 也 不 希望 再 回来 。 


男 一 种 方法 叫 作 “恢复 ”"。 它 意味 看 违例 控制 占有 员 任 来 纠正 当前 的 状 

况 ， 然 后 取得 出 错 的 方法 ， 假 定 下 一 次 会 成 功 执 行 。 大 使 用 恢复 ， 意 味 
者 在 违例 得 到 控制 以 后 仍然 想 继续 执行 。 在 这 种 情况 下 ， 我 们 的 违例 更 
象 一 个 方法 调用 一 一 我 们 用 它 在 Java 中 设置 各 种 各 样 特殊 的 环境 ， 产 生 
类 似 于 “恢复 ”的 行为 〈 换 言 之 ， 此 时 不 是 “ 掷 ? 出 一 个 违例 ， 而 是 调用 一 
个 用 于 解决 问题 的 方法 ) 。 另 外 ， 也 可 以 将 目 己 的 try 块 置 入 一 个 while 
循环 里 ， 用 它 不 断 进 入 try 块 ， 直 到 结果 满意 时 为 止 。 


从 历史 的 角度 看 ， 知 程序 员 使 用 的 操作 系统 文 持 可 恢复 的 违例 控制 ， 基 
终 都 会 用 到 类 似 于 中 断 的 代码 ， 并 跳 过 恢复 进程 。 所 以 尽管 “恢复 ”表面 
上 十 分 不 错 ， 但 在 实际 应 用 中 却 显 得 困难 重重 。 其 中 决定 性 的 原因 可 能 
wes 我 们 的 控制 模块 必须 随时 留意 是 否 产 生 了 违例 ， 以 及 是 人 否 包 含 了 由 
产生 位 置 专用 的 代码 。 这 便 使 代码 很 难 编写 和 维护 一 一 大 型 系统 尤其 如 
此 ， 因 为 违例 可 能 在 多 个 位 置 产 生 。 


9.2.3 违例 规范 


在 Java 中 ， 对 那些 要 调用 方法 的 客户 程序 员 ， 我 们 要 通知 他 们 可 能 从 目 
己 的 方法 里 “ 撕 ” 出 违例 。 这 是 一 种 有 礼貌 的 做 法 ， 只 有 它 才 能 使 客户 程 
序 员 准确 地 知道 要 编写 什么 代码 来 捕获 所 有 洲 在 的 违例 。 当 然 ， 知 你 同 
时 提供 了 源码 ， 客 户 程序 员 甚 至 能 全 盘 检查 代码 ， 找 出 相应 的 throw 语 
人 句 。 但 尽管 如 此 ， 通 常 并 不 随同 源码 提供 库 。 为 解决 这 个 问题 ，Java 提 
供 了 一 种 特殊 的 语法 格式 (并 强迫 我 们 采用 〉， 以 便 礼貌 地 告诉 客户 程 
序 员 该 方法 会 “ 折 ” 出 什么 违例 ， 令 对 方 方 便 地 加 以 控制 。 这 便 是 我 们 在 
这 里 要 讲述 的 “违例 规范 >， 它 属于 方法 声明 的 一 部 分 ， 位 于 目 变量 〈 参 
数 ) 列表 的 后 面 。 


违例 规范 采用 了 一 个 额外 的 关键 字 : throws; 后 面 跟随 全 部 潜在 的 违例 
类 型 。 因 此 ， 我 们 的 方法 定义 看 起 来 应 象 下 面 这 个 样子 : 


void f() throws tooBig, tooSmall, divZero { //... 
FAEH PRAMAS: 
void fQ [ //... 









































它 意 味 着 不 会 从 方法 里 “ 掷 ? 出 违例 〈 除 类 型 为 RuntimeException 的 违例 
以 外 ， 它 可 能 从 任何 地 方 掷 出 一 一 稍 后 还 会 详细 讲述 ) 。 


但 不 能 完全 依赖 违例 规范 一 假若 方法 造成 了 一 个 违例 ， 但 没有 对 其 进 
行 控制 ， 编 译 器 会 侦 测 到 这 个 情况 ， 并 告诉 我 们 必须 控制 违例 ， 或 者 指 
出 应 该 从 方法 里 * 掷 出 一 个 违例 规范 。 通 过 坚持 从 顶部 到 底部 排列 违例 
规范 ，Java 可 在 编译 期 保证 违例 的 正确 性 〈 注 释 @) 。 


D: 这 是 在 C++ 违例 控制 基础 上 一 个 显著 的 进步 ， 后 者 除非 到 运行 期 ， 
这 使 得 C++ 的 违例 控制 机 制 显 得 
用 处 不 大 。 


我 们 在 这 个 地 方 可 采取 欺骗 手段 ， 要 求 “ 搓 "出 一 个 并 没有 发 生 的 违例 。 
编译 器 能 理解 我 们 的 要 求 ， 并 强迫 使 用 这 个 方法 的 用 户 当 作 真 的 产生 了 
那个 违例 处 理 。 在 实际 应 用 中 ， 可 将 其 作为 那个 违例 的 一 个 “ 占 位 符 ” 使 
这 样 一 来 ， 以 后 可 以 方便 地 产生 实际 的 违例 ， 考 需 修改 现 有 的 代 


9.2.4 捕获 所 有 违例 
我 们 可 创建 一 个 控制 器 ， 令 其 捕获 所 有 类 型 的 违例 。 具 体 的 做 法 是 捕获 


基础 类 违例 类 型 Exception (也 存在 其 他 类 型 的 基础 违例 ， 但 Exception 是 
适用 于 几乎 所 有 编程 活动 的 基础 )。 如 下 所 示 : 











catch(Exception e) { 
System.out.printIn("caught an exception"); 
} 


这 段 代码 能 捕获 任何 违例 ， 所 以 在 实际 使 用 时 最 好 将 其 置 于 控制 器 列表 
的 末尾 ， 防 止 跟随 在 后 面 的 任何 特殊 违例 控制 器 失效 。 


对 于 程序 员 和 常用 的 所 有 违例 类 来 说 ， 由 于 Exception 类 是 它们 的 基础 ， 所 
以 我 们 不 会 获得 关于 违例 太 多 的 信息 ， 但 可 调用 来 自 它 的 基础 类 
Throwable 的 方法 : 





String getMessage() 


获得 详细 的 消 妃 。 
String toString() 


A 其 中 包括 详细 的 消 轧 《如果 有 的 
TH) 


void printStackTrace() 
void printStackTrace(PrintStream) 


打印 出 Throwable 和 Throwable 的 调用 堆栈 路 径 。 调 用 堆栈 显示 出 将 我 们 
市 到 违例 发 生地 点 的 方法 调用 的 顺序 。 


第 一 个 版 本 会 打印 出 标准 错误 ， 第 三 个 则 打印 出 我 们 的 选择 流程 。 若 在 
Windows 下 工作 ， 束 不 能 重 定 问 标准 错误 。 因 此 ， 我 们 一 般 愿 意 使 用 第 
二 个 版 本 ， 并 将 结果 送 给 System.out; 这 样 一 来 ， 输 出 就 可 重 定 向 到 我 
们 和 希望 的 任何 路 径 。 


除 此 以 外 ， 我 们 还 可 从 Throwable 的 基础 类 Object〈 所 有 对 象 的 基础 类 
型 ) 获得 另外 一 些 方法 。 对 于 违例 控制 来 说 ， 其 中 一 个 可 能 有 用 的 是 
getClass0， 它 的 作用 是 返回 一 个 对 象 ， 用 它 代 表 这 个 对 象 的 类 。 我 们 可 
依次 用 getName0 或 toString0 碍 询 这 个 Class 类 的 名 字 。 亦 可 对 Class 对 象 
进行 一 些 复杂 的 操作 ， 尽 管 那些 操作 在 违例 控制 中 是 不 必要 的 。 本 章 稍 
后 还 会 详细 讲述 Class 对 象 。 


下 面 是 一 个 特殊 的 例子 ， 它 展示 了 Exception 方 法 的 使 用 〈 若 执行 该 程序 
遇 到 困难 ， 请 参考 第 3 章 3.1.2 小 节 “ 赋 值 >) : 


//: ExceptionMethods. java 











// Demonstrating the Exception Methods 
package c09; 
public class ExceptionMethods { 


public static void main(String[] args) { 


try { 


throw new Exception("Here's my Exception"); 
} catch(Exception e) { 

System.out.println( "Caught Exception"); 
System.out.printin( 

"e.getMessage(): " + e.getMessage()); 
System.out.printin( 

"e.toString(): " + e.toString()); 
System.out.printin("e.printStackTrace():"); 


e.printStackTrace(); 


} 
TTT 


该 程序 输出 如 下 : 


Caught Exception 


e.getMessage(): Here's my Exception 

e.toString(): java.lang.Exception: Here's my Exception 
e.printStackTrace(): 

java.lang.Exception: Here's my Exception 


at ExceptionMethods.main 








可 以 看 到 ， 该 方法 连续 提供 了 大 量 信息 一 一 每 类 信息 都 是 前 一 类 信息 的 





se 
9.2.5 重新 “ 掷 ? 出 违例 


在 东 些 情况 下 ， 我 们 想 重新 扼 出 刚才 产生 过 的 违例 ， 特 别 是 在 用 
Exception 捕 获 所 有 可 能 的 违例 时 。 由 于 我 们 已 拥有 当前 违例 的 句柄 ， 所 
以 只 需 简 单 地 重新 抑 出 那个 句柄 即 可 。 下 面 是 一 个 例子 : 


catch(Exception e) { 
System.out.println(" 一 个 违例 已 经 产生 "); 
throw e; 


} 


重新 “ 掷 ? 出 一 个 违例 导致 违例 进入 更 高 一 级 环境 的 违例 控制 希 中 。 用 于 
同一 个 try 块 的 任何 更 进一步 的 catch 从 名 仍然 会 被 忽略 。 此 外 ， 与 违例 
对 象 有 关 的 所 有 东西 都 会 得 到 保留 ， 所 以 用 于 捕获 特定 违例 类 型 的 更 高 
一 级 的 控制 器 可 以 从 那个 对 象 里 提取 出 所 有 信息 。 


藻 只 是 简单 地 重新 掷 出 当前 违例 ， 我 们 打印 出 来 的 、 与 printStackTrace() 
内 的 那个 违例 有 关 的 信息 会 与 违例 的 起 源 地 对 应 ， 而 不 是 与 重新 掷 出 它 
的 地 点 对 应 。 符 想 安装 新 的 堆栈 跟踪 信息 ， 可 调用 fillInStackTrace()， 它 
会 返回 一 个 特殊 的 违例 对 象 。 这 个 违例 的 创建 过 程 如 下 : 将 当前 堆栈 的 
信息 填充 到 原来 的 违例 对 象 里 。 下 面 列 出 它 的 形式 : 


//: Rethrowing. java 











// Demonstrating fillInStackTrace() 
public class Rethrowing { 
public static void f() throws Exception { 
System.out.printin( 
"Originating the exception in f()"); 


throw new Exception("thrown from f()"); 


} 
public static void g() throws Throwable { 
try { 
f(); 
} catch(Exception e) { 
System.out.printin( 
"Inside g(), e.printStackTrace()"); 
e.printStackTrace(); 
throw e; // 17 


// throw e.fillInStackTrace(); // 18 


} 
} 


public static void 
main(String[] args) throws Throwable { 
try { 


g(); 


} catch(Exception e) { 
System.out.printin( 
"Caught in main, e.printStackTrace()"); 


e.printStackTrace(); 


de 





其 中 最 重要 的 行 写 在 注释 内 标记 出 来 。 注 意 第 17 行 没有 设 为 注释 行 。 它 
的 输出 结果 如 下 : 


originating the exception in f() 


Inside g(), e.printStackTrace() 
java.lang.Exception: thrown from f() 

at Rethrowing.f(Rethrowing. java:8) 

at Rethrowing.g(Rethrowing. java:12) 

at Rethrowing.main(Rethrowing. java: 24) 
Caught in main, e.printStackTrace() 
java.lang.Exception: thrown from f() 

at Rethrowing.f(Rethrowing. java:8) 

at Rethrowing.g(Rethrowing. java:12) 


at Rethrowing.main(Rethrowing. java: 24) 


因此 ， 违 例 堆栈 路 径 无 论 如 何 都 会 记 住 它 的 真正 起 点 ， 无 论 自己 被 重 
SEH? FUEL. 

若 将 第 17 行 标注 《〈 变 成 注释 行 ) ， 而 撤消 对 第 18 行 的 标注 ， 就 会 换 用 
fillInStackTrace(), 445200 F: 





Originating the exception in f() 


Inside g(), e.printStackTrace() 
java.lang.Exception: thrown from f() 


at Rethrowing.f(Rethrowing. java:8) 


at Rethrowing.g(Rethrowing. java:12) 

at Rethrowing.main(Rethrowing. java: 24) 
Caught in main, e.printStackTrace() 
java.lang.Exception: thrown from f() 

at Rethrowing.g(Rethrowing. java:18) 


at Rethrowing.main(Rethrowing. java: 24) 


HF AY fillInStackTrace(), 21877 BY Ay IS Fl AT ELS o 


针对 g() 和 main()，Throwable 类 必须 在 违例 规格 中 出 现 ， 因 为 
fillInStackTrace() 会 生成 一 个 Throwable 对 象 的 句柄 。 由 于 Throwable 是 
Exception 的 一 个 基础 类 ， 上 所 以 有 可 能 获得 一 个 能 够 “ 掷 ? 出 的 对 象 〈 有 共有 
Throwable 属 性 ) ， 但 却 并 非 一 个 Exception 〈 违 例 ) 。 因 此 ， 在 main0 中 
用 于 Exception 的 句柄 可 能 丢失 自己 的 目标 。 为 保证 所 有 东西 均 井 然 有 
序 ， 编 译 器 强制 Throwable 使 用 一 个 违例 规范 。 举 个 例子 来 说 ， 下 述 程 
序 的 违例 便 不 会 在 main() 中 被 捕获 到 : 


//: ThrowOut.java 





public class ThrowOut { 
public static void 
main(String[] args) throws Throwable { 
try { 
throw new Throwable(); 
} catch(Exception e) { 


System.out.println("Caught in main()"); 


} ///:~ 


也 有 可 能 从 一 个 已 经 捕获 的 违例 重新 “ 掷 ? 出 一 个 不 同 的 违例 。 但 假如 这 
样 做 ， 会 得 到 与 使 用 filInStackTrace() 类 似 的 效果 : 与 违例 起 源 地 有 关 的 
言 息 会 全 部 丢失 ， 我 们 留 下 的 是 与 新 的 throw 有 关 的 信息 。 如 下 所 示 : 


//: RethrowNew. java 





// Rethrow a different object from the one that 
// was caught 
public class RethrowNew { 
public static void f() throws Exception { 
System.out.printin( 
"Originating the exception in f()"); 
throw new Exception("thrown from f()"); 
} 
public static void main(String[] args) { 
try { 
f(); 
} catch(Exception e) { 
System.out.println( 
"Caught in main, e.printStackTrace()"); 
e.printStackTrace(); 


throw new NullPointerException("from main"); 


} 
} ///:~ 


输出 如 下 : 


originating the exception in f() 


Caught in main, e.printStackTrace() 
java.lang.Exception: thrown from f() 

at RethrowNew. f(RethrowNew. java: 8) 

at RethrowNew.main(RethrowNew. java:13) 
java.lang.NullPointerException: from main 


at RethrowNew.main(RethrowNew. java:18) 





最 后 一 个 违例 只 知道 自己 来 自 main0， 而 非 来 自 {0。 注 意 Throwable 在 任 
何 违例 规范 中 都 不 是 必需 的 。 


永远 不 必 关 心 如 何 清除 前 一 个 违例 ， 或 者 与 之 有 关 的 其 他 任何 违例 。 它 
和 
动 将 其 清除 。 





9.3 标准 Java 违 例 


Java 包 含 了 一 个 名 为 Throwable 的 类 ， 它 对 可 以 作为 违例 “ 接 ” 出 的 所 有 东 
西 进行 了 描述 。Throwable 对 象 有 两 种 常规 类 型 ( 亦 即 “从 Throwable 继 
IK”) 。 其 中 ，Error 代 表 编 译 期 和 系统 错误 ， 我 们 一 般 不 必 特 意 捕获 它 
们 《 除 在 特殊 情况 以 外 ) 。Exception 是 可 以 从 任何 标准 Java 库 的 类 方法 
中 * 搓 ?出 的 基本 类 型 。 此 外 ， 它 们 亦 可 从 我 们 目 己 的 方法 以 及 运行 期 偶 
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为 获得 违例 的 一 个 综合 概念 ， 最 好 的 方法 是 阅读 由 http://java.sun.com 提 
供 的 联机 Java 文 档 〈 当 然 ， 首 先 下 载 它们 更 好 ) 。 为 了 对 各 种 违例 有 一 
个 大 概 的 印象 ， 这 个 工作 是 相当 有 价值 的 。 但 大 家 不 久 就 会 发 现 ， 除 名 
字 外 ， 一 个 违例 和 下 一 个 违例 之 间 并 不 存在 任何 特殊 的 地 方 。 此 外 ， 

Java 提 供 的 违例 数量 正在 日 益 增 多 ;从 本 质 上 说 ， 把 它们 印 到 一 本 书 里 
是 没有 意义 的 。 大 家 从 其 他 地 方 获得 的 任何 新 库 可 能 也 提供 了 它们 自己 
的 违例 。 我 们 最 需要 掌握 的 是 基本 概念 ， 以 及 用 这 些 违 例 能 够 做 什么 。 








java.lang.Exception 

这 是 程序 能 捕获 的 基本 违例 。 其 他 违例 都 是 从 它 衍 生出 去 的 。 这 里 要 注 
意 的 是 违例 的 名 字 代 表 发 生 的 问题 ， 而 且 违 例 名 通常 都 是 精心 挑选 的 ， 

可 以 很 清楚 地 说 明 到 底 发 生 了 什么 事情 。 违 例 并 不 全 是 在 java.lang 中 定 
SU; 有 些 是 T 如 util，net 以 及 io 等 一 一 我 们 


可 以 从 它们 的 完整 类 名 中 看 出 这 一 点 ， 或 者 观察 它们 从 什么 继承 。 例 
如 ， 所 有 IO 违例 都 是 从 java.io. ceed 的 。 


9.3.1 RuntimeException 的 特殊 情况 
本 章 的 第 一 个 例子 是 : 


if(t == null) 





throw new NullPointerException(); 


看 起 来 似乎 在 传递 进入 一 个 方法 的 每 个 句柄 中 都 必须 检查 null CAS 
知道 调用 者 是 否 已 传递 了 一 个 有 效 的 句柄 ) ， 这 无 疑 是 相当 可 怕 的 。 但 
幸运 的 是 ， 我 们 根本 不 必 这 样 做 一 一 它 属于 Java 进 行 的 标准 运行 期 检查 











的 一 部 分 。 知 对 一 个 空 句柄 发 出 了 调用 ，Java 会 自动 产生 一 个 
NullPointerException 违 例 。 所 以 上 述 代码 在 任何 情况 下 都 是 多 余 的 。 


这 个 类 别 里 含有 一 系列 违例 类 型 。 它 们 全 部 由 Java 自 动 生成 ， 毋 需 我 们 
杀 上 自动 手 把 它 们 包含 到 自己 的 违例 规范 里 。 最 方便 的 是 ， 通 过 将 它们 置 
入 单独 一 个 名 为 RuntimeException 的 基础 类 下 面 ， 它 们 全 部 组 合 到 一 
起 。 这 是 一 个 很 好 的 继承 例子 : 它 建 立 了 一 系列 具有 某 种 共通 性 的 类 
型 ， 都 具有 某 些 共通 的 特征 与 行为 。 此 外 ， 我 们 没 必要 专门 写 一 个 违例 
规范 ， 指 出 一 个 方法 可 能 会 “ 掷 ? 出 一 个 RuntimeException， 因 为 已 经 假 
定 可 能 出 现 那 种 情况 。 由 于 它们 用 于 指出 编程 中 的 错误 ， 所 以 几乎 永远 
不 必 专 门 捕 获 一 个 “运行 期 违例 ”RuntimeException 一 一 它 在 默认 情 
况 下 会 自动 得 到 处 理 。 若 必须 检查 RuntimeException， 我 们 的 代码 就 会 
变 得 相当 繁复 。 在 我 们 自己 的 包 里 ， 可 选择 “ 接 ” 出 一 部 分 
RuntimeException. 

如 果 不 捕获 这 些 违例 ， 又 会 出 现 什么 情况 呢 ? 由 于 编译 器 并 不 强制 违例 
规范 捕获 它们 ， 所 以 假如 不 捕获 的 话 ， 一 个 RuntimeException 可 能 过 滤 
卸 我 们 到 达 main0) 方 法 的 所 有 途径 。 为 体会 此 时 发 生 的 事情 ， 请 试 试 下 
面 这 个 例子 : 


//: NeverCaught.java 











// Ignoring RuntimeExceptions 
public class NeverCaught { 
static void f() { 
throw new RuntimeException("From f()"); 
} 
static void g() { 
f(); 
} 


public static void main(String[] args) { 


g(); 
} 
} ///:~ 


大 家 已 经 看 到 ， 一 个 RuntimeException 〈 或 者 从 它 继承 的 任何 东西 ) 属 
于 一 种 特殊 情况 ， 因 为 编译 器 不 要 求 为 这 些 类 型 指定 违例 规范 。 


输出 如 下 : 

java.lang.RuntimeException: From f() 

at NeverCaught.f(NeverCaught.java:9) 

at NeverCaught.g(NeverCaught.java:12) 

at NeverCaught.main(NeverCaught.java:15) 


所 以 答案 就 是 : 假若 一 个 RuntimeException 获 得 到 达 main() 的 所 有 途 
同时 不 被 捕获 ， 那 么 当 程 序 退 出 时 ， 会 为 那个 违例 调用 
printStackTrace(). 


注意 也 许 能 在 自己 的 代码 中 仪 忽略 RuntimeException， 因 为 编译 右 已 正 
确 实行 了 其 他 所 有 控制 。 因 为 RuntimeException 在 此 时 代表 一 个 编程 错 
VR: 

(1) 一 个 我 们 不 能 捕获 的 错误 《〈 例 如， 由 客户 程序 员 接 收 传递 给 自己 方 
法 的 一 个 空 句 柄 ) 。 


(2) 作为 一 名 程序 员 ， 一 个 应 在 自己 的 代码 中 检查 的 错误 如 
ArrayIndexOutOfBoundException， 此 时 应 注意 数组 的 大 小 ) 。 


Tadd 最 好 的 做 法 是 在 这 种 情况 下 违例 ， 因 为 它们 有 助 于 程序 的 调 
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另外 一 个 有 趣 的 地 方 是 ， 我 们 不 可 将 Java 违 例 划 分 为 单一 用 途 的 工具 。 
的 确 ， 它 们 设计 用 于 控制 那些 讨 大 的 运行 期 错误 一 一 由 代码 控制 范围 之 
外 的 其 他 力量 产生 。 但 是 ， 它 也 特别 有 助 于 调试 菜 些 特殊 类 型 的 编程 错 





误 ， 那 些 是 编译 器 侦 测 不 到 的 。 


9.4 创建 目 己 的 违例 


并 不 一 定 非 要 使 用 Java 违 例 。 这 一 点 必须 掌握 ， 因 为 经 常 都 需要 创建 自 
己 的 违例 ， 以 便 指出 自己 的 库 可 能 生成 的 一 个 特殊 错误 一 一 但 创建 Java 
分 级 结构 的 时 候 ， 这 个 错误 是 无 法 预知 的 。 
为 创建 自己 的 违例 类 ， 必 须 从 一 个 现 有 的 违例 类 型 继承 一 一 最 好 在 含义 
上 与 新 违例 近似 。 继 承 一 个 违例 相当 简单 : 


//: Inheriting.java 





// Inheriting your own exceptions 
class MyException extends Exception { 
public MyException() {} 
public MyException(String msg) { 


super(msg); 


} 
public class Inheriting { 
public static void f() throws MyException { 
System. out.printin( 
"Throwing MyException from f()"); 
throw new MyException(); 
} 
public static void g() throws MyException { 


System.out.printin( 


"Throwing MyException from g()"); 
throw new MyException("Originated in g()"); 
} 
public static void main(String[] args) { 
try { 
f(); 
} catch(MyException e) { 
e.printStackTrace(); 
} 
try { 


g(); 


} catch(MyException e) { 


e.printStackTrace(); 


} 
下 


继承 在 创建 新 类 时 发 生 : 


class MyException extends Exception { 


public MyException() {} 
public MyException(String msg) { 


super(msg); 


这 里 的 关键 是 “extends Exception”, CMR: 除 包括 一 个 Exception 
的 全 部 含义 以 外 ， 还 有 更 多 的 含义 。 增 加 的 代码 数 量 非常 少 实际 只 
添加 了 两 个 构建 器 ， 对 MyException 的 创建 方式 进行 了 定义 。 请 记 住 ， 
假如 我 们 不 明确 调用 一 个 基础 类 构建 器 ， 编 译 器 会 自动 调用 基础 类 默认 
构建 器 。 在 第 二 个 构建 器 中 ， 通 过 使 用 super 关 键 字 ， 明 确 调用 了 带 有 一 
个 String 参 数 的 基础 类 构建 器 。 


该 程序 输出 结果 如 下 : 


Throwing MyException from f() 








MyException 

at Inheriting.f(Inheriting.java:16) 

at Inheriting.main(Inheriting.java:24) 
Throwing MyException from g() 
MyException: Originated in g() 

at Inheriting.g(Inheriting.java:20) 


at Inheriting.main(Inheriting.java:29) 


可 以 看 到 ， 在 从 f0* 括 ?出 的 MyException 违 例 中 ， 缺 乏 详 细 的 消息 。 


违例 时 ， 还 可 以 采取 更 多 的 操作 。 我 们 可 添加 额外 的 构建 器 
及 成 员 ; 


//: Inheriting2.java 





// Inheriting your own exceptions 


class MyException2 extends Exception { 
public MyException2() {} 
public MyException2(String msg) { 
super(msg); 
} 
public MyException2(String msg, int x) { 


super(msg); 


public int val() { return i; } 
private int 1; 
} 
public class Inheriting2 { 
public static void f() throws MyException2 { 
System.out.printin( 
"Throwing MyException2 from f()"); 
throw new MyException2(); 
} 
public static void g() throws MyException2 { 
System.out.println( 
"Throwing MyException2 from g()"); 


throw new MyException2("Originated in g()"); 


public static void h() throws MyException2 { 
System.out.printin( 
"Throwing MyException2 from h()"); 
throw new MyException2( 
"Originated in h()", 47); 
} 
public static void main(String[] args) { 
try { 
f(); 
} catch(MyException2 e) { 
e.printStackTrace(); 
} 
try { 


g(); 


} catch(MyException2 e) { 
e.printStackTrace(); 

} 

try { 
h(); 

} catch(MyException2 e) { 
e.printStackTrace(); 


System.out.printin("e.val() = " + e.val()); 


} 
} ///:~ 





此 时 添加 了 一 个 数据 成 员 i， 同时 添 加 了 一 个 特殊 的 方法 ， 用 它 读 取 那 
个 值 ， 也 添加 了 一 个 额外 的 构建 融 ， 用 它 设置 那个 值 。 输 出 结果 如 下 : 


Throwing MyException2 from f() 


MyException2 

at Inheriting2.f(Inheriting2.java:22) 

at Inheriting2.main(Inheriting2.java:34) 
Throwing MyException2 from g() 
MyException2: Originated in g() 

at Inheriting2.g(Inheriting2.java:26) 

at Inheriting2.main(Inheriting2. java: 39) 
Throwing MyException2 from h() 
MyException2: Originated in h() 

at Inheriting2.h(Inheriting2.java:30) 

at Inheriting2.main(Inheriting2. java: 44) 


e.val() = 47 


由 于 违例 不 过 是 另 一 种 形式 的 对 象 ， 所 以 可 以 继续 这 个 进程 ， 进 一 步 增 
强 违例 类 的 能 力 。 但 要 注意 ， 对 使 用 目 己 这 个 包 的 客户 程序 员 来 说 ， 他 
们 可 能 错过 所 有 这 些 增强 。 因 为 他 们 可 能 只 是 简单 地 寻找 准备 生成 的 违 
例 ， 除 此 以 外 不 做 任何 事情 一 一 这 是 大 多 数 Java 库 违例 的 标准 用 法 。 寿 
人 











//: SimpleException.java 

class SimpleException extends Exception { 

pies 

它 要 依赖 编译 器 来 创建 默认 构建 器 (会 自动 调用 基础 类 的 默认 构建 


fit) 。 当 然 ， 在 这 种 情况 下 ， 我 们 不 会 得 到 一 个 SimpleException(String) 
构建 器 ， 但 它 实 际 上 也 不 会 经 常用 到 。 


9.5 违例 的 限制 


履 盖 一 个 方法 时 ， 只 能 产生 已 在 方法 的 基础 类 版 本 中 定义 的 违例 。 这 是 
一 个 重要 的 限制 ， 因 为 它 意 味 着 与 基础 类 协同 工作 的 代码 也 会 自动 应 用 
0 ee ee vee 
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下 面 这 个 例子 演示 了 强加 在 违例 身上 的 限制 类型 “在 编译 期 ) : 


//: StormyInning. java 











// Overridden methods may throw only the 
// exceptions specified in their base-class 
// versions, or exceptions derived from the 
// base-class exceptions. 
Class BaseballException extends Exception {} 
class Foul extends BaseballException {} 
class Strike extends BaseballException {} 
abstract class Inning { 

Inning() throws BaseballException {} 

void event () throws BaseballException { 

// Doesn't actually have to throw anything 

} 

abstract void atBat() throws Strike, Foul; 


void walk() {} // Throws nothing 


class StormException extends Exception {} 
class RainedOut extends StormException {} 
class PopFoul extends Foul {} 
interface Storm { 

void event() throws RainedOut; 

void rainHard() throws RainedOut; 
} 
public class StormyInning extends Inning 

implements Storm { 

// OK to add new exceptions for constructors, 
// but you must deal with the base constructor 
// exceptions: 

StormyInning() throws RainedOut, 

BaseballException {} 

StormyInning(String s) throws Foul, 

BaseballException {} 

// Regular methods must conform to base class: 
//! void walk() throws PopFoul {} //Compile error 
// Interface CANNOT add exceptions to existing 
// methods from the base class: 

//! public void event() throws RainedOut {} 
// If the method doesn't already exist in the 


// base class, the exception is OK: 


public void rainHard() throws RainedOut {} 
// You can choose to not throw any exceptions, 
// even if base version does: 
public void event() {} 
// Overridden methods can throw 
// inherited exceptions: 
void atBat() throws PopFoul {} 
public static void main(String[] args) { 
try { 
StormyInning si = new StormyInning(); 
Si.atBat(); 
} catch(PopFoul e) { 
} catch(RainedOut e) { 
} catch(BaseballException e) {} 
// Strike not thrown in derived version. 
try { 
// What happens if you upcast? 
Inning i = new StormyInning(); 
1.atBat(); 
// You must catch the exceptions from the 
// base-class version of the method: 
} catch(Strike e) { 


} catch(Foul e) { 


} catch(RainedOut e) { 
} catch(BaseballException e) {} 


} 
A f/f 
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一 小 违例 ,得 它们 实际 上 没有 那 桩 做 。 这 是 合法 的 ， 因 为 它 a 
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也 适用 于 abstract 3, 就 象 在 atBat(0) 里 展示 的 那样 。 


“interface Storm” 非 常 有 趣 ， 因 为 它 包含 了 在 Incoming 中 定义 的 一 个 方法 

event()， 以 及 不 是 在 其 中 定义 的 一 个 方法 。 这 两 个 方法 都 会 “ 接 ” 出 

一 个 新 的 违例 类 型 RainedOut。 当 执行 到 “StormyInning 

extends’ ' 和 “implements Storm” 的 时 候 ， 可 以 看 到 Storm 中 的 eventO 方 法 不 
能 改变 Inning 中 的 event0 的 违例 接口 。 同样 地 ， 这 种 设计 是 十 分 合理 

的 ; 否则 的 话 ， 当 我 们 操作 基础 类 时 ， 便 根本 无 法 知道 自己 捕获 的 是 

正确 的 东西 。 当 然 ， 假 如 interface 中 定义 的 一 个 方法 不 在 基础 类 里 ， vi 

如 rainHardO0， 它 产生 违例 时 就 没什么 问题 。 


对 违例 的 限制 并 不 适用 于 构建 问 。 在 StormyInning 中 ， 我 们 可 看 到 一 个 
构建 器 能 够 “ 掷 ? 出 它 希 望 的 任何 东西 ， 无 论 基 础 类 构建 器 “ 掷 ? 出 什么 。 

然而 ， 由 于 必须 坚持 按 某 种 方式 调用 基础 类 构建 器 (在 这 里 ， 会 自动 调 
用 默认 构建 器 〉 ， 所 以 衍生 类 构建 器 必须 在 上 自己 的 违例 规范 中 声明 所 有 
基础 类 构建 器 违 B. 


StormyInning.walk() PR iE I EA E E R T eE, TT 
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调用 Inning.walk0， 而 且 它 不 必 控 制 任何 违例 。 但 在 以 后 蔡 换 从 Inning 衍 
生 的 一 个 类 的 对 象 时 ， 违 例 就 会 “ 掷 ? 出 ， 造 成 代码 执行 的 中 断 。 通 过 强 
迫 衍 生 类 方法 遵守 基础 类 方法 的 违例 规范 ， 对 象 的 蔡 换 可 保持 连贯 性 。 


敢 盖 过 的 event(0) 方 法 加 我 们 显示 出 一 个 方法 的 衍生 类 版 本 可 以 不 产生 任 
何 违例 即便 基础 类 版 本 要 产生 违例 。 同 样 地 ， 这 样 做 是 必要 的 ， 因 
为 它 不 会 中 断 那些 已 假定 基础 类 版 本 会 产生 违例 的 代码 。 差 不 多 的 道理 
亦 适 用 于 atBat0， 它 会 “ 扼 ” 出 PopFoul 从 Foul 衍 生出 来 的 一 个 违例 ， 




















而 Foul 违 例 古 由 atBat0) 的 基础 类 版 本 产生 的 。 这 样 一 来 ， 假 如 有 人 在 上 自 
己 的 代码 里 操作 Inning， 同 时 调用 了 atBat()， 就 必须 捕获 Foul 违 例 。 由 
于 PopFou 是 从 Fou 衍 生 的 ， 所 以 违例 控制 器 《模块 ) 也 会 捕获 
PopFoul. 


最 后 一 个 有 趣 的 地 方 在 main0 内 部 。 在 这 个 地 方 ， 假 如 我 们 明确 操作 一 
个 StormyInning 对 象 ， 编 译 器 丈 会 强迫 我 们 只 捕获 特定 于 那个 类 的 违 
例 。 但 假如 我 们 上 济 造 型 到 基础 类 型 ， 编 译 器 就 会 强迫 我 们 捕获 针对 基 
础 类 的 违例 。 通 过 所 有 这 些 限 制 ， 违 例 控制 代码 的 “健壮 ”程度 获得 了 大 
幅度 改善 (注释 @))。 


®©: ANSI/ISO “C++ 施加 了 类 似 的 限制 ， 要 求 衍 生 方 法 违例 与 基础 类 方 
法 搓 出 的 违例 相同 ， 或 者 从 后 者 衍生 。 在 这 种 情况 下 ，C++ 实 际 上 能 够 
在 编译 期 间 检查 违例 规范 。 


我 们 必须 认识 到 这 一 点 : 尽管 违例 规范 是 由 编译 费 在 继承 期 间 强 行 革 守 
的 ， 但 违例 规范 并 不 属于 方法 类 型 的 一 部 分 ， 后 者 仅 包 括 了 方法 名 以 及 
目 变 量 类 型 。 因 此 ， 我 们 不 可 在 违例 规范 的 基础 上 和 宪 冀 方法 。 除 此 以 
外 ， 尺 管 违例 规范 存在 于 一 个 方法 的 基础 类 版 本 中 ， 但 并 不 表示 它 必须 
在 方法 的 衍生 类 版 本 中 存在 。 这 与 方法 的 “继承 ” 左 有 不 同 ( 进 行 继 承 
时 ， 基 础 类 中 的 方法 也 必须 在 衍生 关中 存在 ) 。 换 言 之 ， 用 于 一 个 特定 
方法 的 “违例 规范 接口 ”可 能 在 继承 和 禾 新 时 变 得 更 “ 窄 ”， 但 它 不 会 变 得 
更 “ 宽 ” 一 一 这 与 继承 时 的 类 接口 规则 是 正好 相反 的 。 


























9.6 用 finally 清 除 


无 论 一 个 违例 是 否 在 try 块 中 发 生 ， 我 们 经 常 都 想 执 行 一 些 特定 的 代码 。 
对 一 些 特定 的 操作 ， 经 常 都 会 遇 到 这 种 情况 ， 但 在 恢复 内 存 时 一 般 部 不 
需要 (因为 垃圾 收集 占 会 自动 照料 一 切 )。 为 达到 这 个 目的 ， 可 在 所 有 
违例 控制 器 的 末尾 使 用 一 个 finally 从 名 注释 ) 。 所 以 完整 的 违例 控 
制 小 节 象 下 面 这 个 样子 : 





try { 

/ 要 保卫 的 区 域 : 

/ 可 能 “ 掷 ?出 A,B, 或 C 的 危险 情况 
} catch (A al) { 

/ 控制 器 A 

} catch (B b1) { 

/ 控制 器 B 

} catch (C c1) { 

/ 控制 器 C 

} finally { 

/每 次 都 会 发 生 的 情况 
} 
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为 演示 finally 从 句 ， 请 试验 下 面 这 个 程序 : 


//: Finallyworks.java 


// The finally clause is always executed 
public class Finallyworks { 
static int count = 0; 
public static void main(String[] args) { 
while(true) { 
try { 
// post-increment is zero first time: 
if(count++ == 0) 
throw new Exception(); 
System.out.printin("No exception"); 
} catch(Exception e) { 
System.out.printin("Exception thrown"); 
} finally { 
System.out.printin("in finally clause"); 


if(count == 2) break; // out of "while" 


} 
} ///:~ 


通过 该 程序 ， 我 们 亦 可 知道 如 何 应 付 Java 违 例 〈 类 似 C++ 的 违例 ) 不 允 
许 我 们 恢复 至 违例 产生 地 方 的 这 一 事实 。 知 将 上 自己 的 ty 块 置 入 一 个 循环 
内 ， 就 可 建立 一 个 条 件 ， 它 必须 在 继续 程序 之 前 满足 。 亦 可 添加 一 个 








static 计 数 占 或 者 男 一 些 设 备 ， 人 允许 循环 在 放弃 以 前 笃 试 数 种 不 同 的 方 
法 。 这 样 一 来 ， 我 们 的 程序 可 以 变 得 更 加 “健壮 ”。 


输出 如 下 : 


Exception thrown 


in finally clause 
No exception 


in finally clause 


无 论 是 否 “ 搓 ?出 一 个 违例 ，finally 从 名 都 会 执行 。 
9.6.1 用 finally 做 什么 


在 没有 “垃圾 收集 ”以 及 “自动 调用 破坏 器 ”机 制 的 一 种 语言 中 注释 
©) ，finally 显 得 特别 重要 ， 因 为 程序 员 可 用 它 担 保 内 存 的 正确 释放 
一 一 无 论 在 try 块 内 部 发 生 了 什么 状况 。 但 Java 提 供 了 垃圾 收集 机 制 ， 所 
以 内 存 的 释放 几乎 绝对 不 会 成 为 问题 。 男 外 ， 它 也 没有 构建 器 可 供 调 
用 。 既 然 如 此 ，Java 里 何 时 才 会 用 到 finally 呢 ? 


©): “破坏 器 ”(Destructor) 是 “构建 器 ”(Constructor) 的 反义词 。 它 代 
表 一 个 特殊 的 函数 ， 一 旦 某 个 对 象 失去 用 处 ， 通 第 就 会 调用 它 。 我 们 肯 
定 知 道 在 哪里 以 及 何 时 调用 破坏 器 。C++ 提 供 了 自动 的 破坏 器 调用 机 
制 ， 但 Delphi 的 Object ”Pascal 版 本 1 及 2 却 不 具备 这 一 能 力 〈 在 这 种 语言 
中 ， 破 坏 器 的 含义 与 用 法 都 发 生 了 变化 ) 。 


除 将 内 存 设 回 原始 状态 以 外 ， 各 要 设置 妃 一 些 东 西 ，finally 就 是 必需 
的 。 例 如 ， 我 们 有 时 需要 打开 一 个 文件 或 者 建立 一 个 网 络 连接 ， 或 者 在 
屏 妖 上 男 一 些 东 西 ， 其 至 设置 外 部 世界 的 一 个 开关， 等 等 。 如 下 例 所 
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//: OnOffSwitch.java 


// Why use finally? 


class Switch { 
boolean state = false; 
boolean read() { return state; } 
void on() { state = true; } 
void off() { state = false; } 
} 
public class OnOffSwitch { 
static Switch sw = new Switch(); 
public static void main(String[] args) { 
try { 
sw.on(); 
// Code that can throw exceptions... 
sw.off(); 

} catch(NullPointerException e) { 
System.out.printin("NullPointerException"); 
sw.off(); 

} catch(IllegalArgumentException e) { 
System.out.printin( "IOException" ); 


sw.off(); 


} ///:~ 


这 里 的 目标 是 保证 main() 完 成 时 开关 处 于 关闭 状态 ， 所 以 将 sw.off() 置 于 
try 块 以 及 每 个 违例 控制 器 的 末尾 。 但 产生 的 一 个 违例 有 可 能 不 是 在 这 里 
捕获 的 ， 这 便 会 错过 sw.offO。 然 而 ， 利 用 finally， 我 们 可 以 将 来 自 try 块 
的 关闭 代码 只 置 于 一 个 地 方 : 


//: WithFinally.java 





// Finally Guarantees cleanup 
class Switch2 { 
boolean state = false; 
boolean read() { return state; } 
void on() { state = true; } 
void off() { state = false; } 
} 
public class WithFinally { 
static Switch2 sw = new Switch2(); 
public static void main(String[] args) { 
try { 
sw.on(); 
// Code that can throw exceptions... 
} catch(NullPointerException e) { 
System.out.printin("NullPointerException" ); 
} catch(IllegalArgumentException e) { 
System.out.printin( "IOException" ); 


} finally { 


sw.off(); 


} 
} ///:~ 


在 这 儿 ，sw.off0 己 移 至 一 个 地 方 。 无 论 发 生 什么 事情 ， 都 肯定 会 运行 
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即使 违例 不 在 当前 的 catch 从 句 集 里 捕获 ，finally 都 会 在 违例 控制 机 制 转 
到 更 高 级 别 搜索 一 个 控制 器 之 前 得 以 执行 。 如 下 所 示 : 


//: AlwaysFinally.java 


// Finally is always executed 
class Ex extends Exception {} 
public class AlwaysFinally { 
public static void main(String[] args) { 
System. out.printin( 
"Entering first try block"); 
try { 
System.out.printin( 
"Entering second try block"); 
try { 
throw new Ex(); 
} finally { 


System.out.printin( 


"finally in 2nd try block"); 
} 
} catch(Ex e) { 
System.out.println( 
"Caught Ex in first try block"); 
} finally { 
System.out.println( 


"finally in ist try block"); 


} 
} ///:~ 


该 程序 的 输出 展示 了 其 体 发 生 的 事情 : 


Entering first try block 


Entering second try block 
finally in 2nd try block 
Caught Ex in first try block 


finally in ist try block 


知 调 用 了 break 和 continue 语 句 ，finally 语 句 也 会 得 以 执行 。 请 注意 ， 与 
作 上 标签 的 break 和 continue 一 道 ，finally 排 除了 Java 对 goto 跳 转 语 句 的 需 


9.6.2 缺点 : 丢失 的 违例 


一 般 情况 下 ，Java 的 违例 实施 方案 都 显得 十 分 出 色 。 不 幸 的 是 ， 它 依然 
存在 一 个 缺点 。 尽 管 违例 指出 程序 里 存在 一 个 危机 ， 而 且 绝 不 应 忽略 ， 
但 一 个 违例 仍 有 可 能 简单 地 “丢失 ”。 在 采用 finally 从 句 的 一 种 特殊 配置 
下 ， 便 有 可 能 发 生 这 种 情况 : 


//: LostMessage.java 


// How an exception can be lost 
class VeryImportantException extends Exception { 
public String toString() { 


return "A very important exception!"; 


} 


class HoHumException extends Exception { 
public String toString() { 


return "A trivial exception"; 


j 


public class LostMessage { 
void f() throws VeryImportantException { 
throw new VeryImportantException(); 
} 
void dispose() throws HoHumException { 


throw new HoHumException(); 


public static void main(String[] args) 
throws Exception { 
LostMessage lm = new LostMessage(); 
try { 
1m. f(); 
} finally { 


1lm.dispose(); 


} 

} 
VALLES 
输出 如 下 : 


A trivial exception 


at LostMessage.dispose(LostMessage.java:21) 


at LostMessage.main(LostMessage.java:29) 


可 以 看 到 ， 这 里 不 存在 VeryImportantException (非常 重要 的 违例 ) 的 迹 
象 ， 它 只 是 简单 地 被 finally 从 句 中 的 HoHumException 代 奉 了 。 


这 是 一 项 相当 严重 的 缺陷 ， 因 为 它 意味 着 一 个 违例 可 能 完全 丢失 。 而 且 
就 象 前 例 演示 的 那样 ， 这 种 丢失 显得 非常 “自然 "， 很 难 被 人 查 出 蛛 丝 马 
迹 。 而 与 此 相反 ，C++ 里 如 果 第 二 个 违例 在 第 一 个 违例 得 到 控制 前 产 
生 ， 就 会 被 当 作 一 个 严重 的 编程 错误 处 理 。 或 许 Java 以 后 的 版 本 会 纠正 
这 个 问题 〈 上 述 结 果 是 用 Java 1.1 生 成 的 ) 。 








9.7 构建 硕 


为 违例 编写 代码 时 ， 我 们 经 常 要 解决 的 一 个 问题 是 :“ 一 旦 产生 违例 ， 

会 正确 地 进行 清除 吗 ? ”大 多 数 时 候 都 会 非常 安全 ， 但 在 构建 保 中 却 古 
一 个 大 问题 。 构 建 锅 将 对 象 置 于 一 个 安全 的 起 始 状态 ， 但 它 可 能 执行 一 
些 操作 一 一 如 打开 一 个 文件 。 除 非 用 户 完 成 对 象 的 使 用 ， 并 调用 一 个 特 
殊 的 清除 方法 ， 人 否则 那些 操作 不 会 得 到 正确 的 清除 。 奉 从 一 个 构建 融 内 
部 “ 掷 ? 出 一 个 违例 ， 这 些 清 除 行为 也 可 能 不 会 正确 地 发 生 。 所 有 这 些 都 
意味 着 在 编写 构建 器 时 ， 我 们 必须 特别 加 以 留意 。 


由 于 前 面 刚 学 了 finaly， 所 以 大 家 可 能 认为 它 是 一 种 合适 的 方案 。 但 事 
情 并 没有 这 么 简单 ， 因 为 finally 每 次 都 会 执行 清除 代码 一 一 即使 我 们 在 
清除 方法 运行 之 前 不 想 执 行 清 除 代 码 。 因 此 ， 假 如 真 的 用 finally 进 行 清 
除 ， 必 须 在 构建 器 正常 结束 时 设置 某 种 形式 的 标志 。 而 且 只 要 设置 了 标 
志 ， 束 不 要 执行 finally 块 内 的 任何 东西 。 由 于 这 种 做 法 并 不 完美 (需要 
将 一 个 地 方 的 代码 同 男 一 个 地 方 的 结合 起 来 )》， 所 以 除非 特别 需要 ， 合 
则 一 般 不 要 尝试 在 finally 中 进行 这 种 形式 的 清除 。 


在 下 面 这 个 例子 里 ， 我 们 创建 了 一 个 名 为 InputFile 的 类 。 它 的 作用 是 打 

开 一 个 文件 ， 然 后 每 次 读 取 它 的 一 行内 容 〈 转 换 为 一 个 字 串 ) 。 它 利用 

了 由 Java 标 准 IO 库 提供 的 FileReader 以 及 BufferedReader 类 (将 于 第 10 章 

和 
法 : 


//: Cleanup.java 


























// Paying attention to exceptions 
// in constructors 
import java.io.*; 
class InputFile { 
private BufferedReader in; 


InputFile(String fname) throws Exception { 


try { 
in = 
new BufferedReader ( 
new FileReader(fname) ); 
// Other code that might throw exceptions 
} catch(FileNotFoundException e) { 
System. out.printin( 
"Could not open " + fname); 
// Wasn't open, so don't close it 
throw e; 
} catch(Exception e) { 
// All other exceptions must close it 
try { 
in.close(); 
} catch(IOException e2) { 
System.out.printin( 
"in.close() unsuccessful"); 
} 
throw e; 
} finally { 


// Don't close it here!!! 


String getLine() { 

String s; 

try { 
S = in.readLine(); 

} catch(IOException e) { 
System.out.printin( 

"readLine() unsuccessful"); 

s = "failed"; 

} 


return s; 


} 


void cleanup() { 
try { 
in.close(); 
} catch(IOException e2) { 
System.out.printin( 


"in.close() unsuccessful"); 


} 


public class Cleanup { 


public static void main(String[] args) { 


try { 


InputFile in = 
new InputFile("Cleanup.java"); 

String s; 

int i = 1; 

while((s = in.getLine()) != null) 
System.out.printin(""+ i++ +": " + s); 


in.cleanup(); 


(aye 


catch(Exception e) { 
System.out.printin( 
"Caught in main, e.printStackTrace()"); 


e.printStackTrace(); 


} 
} ///:~ 


该 例 使 用 了 Java 1.1 IO 类 。 


用 于 ImputFile 的 构建 器 采用 了 一 个 String CFB) 参数 ， 它 代表 我 们 想 打 
开 的 那个 文件 的 名 字 。 在 一 个 try 块 内 部 ， 它 用 该 文件 名 创建 了 一 个 
FileReader。 对 FileReader 来 说 ， 除 非 转 移 并 用 它 创 建 一 个 能 够 实 际 5 

之 “交谈 ”的 BufferedReader， 否 则 便 没 什么 用 处 。 注 意 InputFile 的 一 个 好 
处 就 是 它 同 时 合并 了 这 两 种 行动 。 


看 FileReader 构 建 问 不 成 功 ， 就 会 产生 一 个 FileNotFoundException (文件 
未 找到 违例 ) 。 必 须 单 独 捕获 这 个 违例 一 一 这 属于 我 们 不 想 关 闭 文件 的 
一 种 特殊 情况 ， 因 为 文件 尚未 成 功 打开 。 其 他 任何 捕获 从 名 Catch) 都 
必须 关闭 文件 ， 因 为 文件 已 在 进入 那些 捕获 从 句 时 打开 (当然 ， 如 果 多 
个 方法 都 能 产生 一 个 FileNotFoundException 违 例 ， 就 需要 稍微 用 一 些 技 














巧 。 此 时 ， 我 们 可 将 不 同 的 情况 分 隔 到 数 个 try 块 内 ) 。close() 方 法 会 迫 
出 一 个 尝试 过 的 违例 。 即 使 它 在 另 一 个 catch 从 名 的 代码 块 内 ， 该 违例 也 
会 得 以 捕获 一 一 对 Java 编 译 器 来 说 ， 那 个 catch 从 句 不 过 是 另 一 对 花 括号 
而 已 。 执 行 完 本 地 操作 后 ， 违 例会 被 重新 “ 掷 ?出 。 这 样 做 是 必要 的 ， 因 
为 这 个 构建 器 的 执行 已 经 失败 ， 我 们 不 希望 调用 方法 来 假设 对 象 已 正确 
创建 以 及 有 效 。 


在 这 个 例子 中 ， 没 有 采用 前 述 的 标志 技术 ，finally 从 名 显然 不 是 关闭 文 

件 的 正确 地 方 ， 因 为 这 可 能 在 每 次 构建 费 结 束 的 时 候 关 闭 它 。 由 于 我 们 
eee 象 处 于 活动 状态 时 一 直 保持 打开 状态 ， 所 以 这 样 
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getLine() 方 法 会 返回 一 个 字 串 ， 其 中 包含 了 文件 中 下 一 行 的 内 容 。 它 调 
用 了 readLine()， 后 者 可 能 产生 一 个 违例 ， 但 那个 违例 会 被 捕获 ， 使 
getLine() 不 会 再 产生 任何 违例 。 对 违例 来 说 ， 一 项 特别 的 设计 问题 是 决 
定 在 这 一 级 完全 控制 一 个 违例 ， 还 是 进行 部 分 控制 ， 并 传递 相同 〈 或 不 
ED 的 违例 ， 或 者 只 是 简单 地 传递 它 。 在 适当 的 时 候 ， 简 单 地 传递 可 极 
大 简化 我 们 的 编码 工作 。getLine() 方 法 会 变 成 : 





String getLine() throws IOException { 

return in.readLine(); 

} 

但 是 当然 ， 调 用 者 现在 需要 对 可 能 产生 的 任何 IOException 进 行 控制 。 


用 户 使 用 完毕 InputFile 对 象 后 ， 必 须 调用 cleanup0 方 法 ， 以 便 释 放 由 
BufferedReader 以 及 / 或 者 FileReader 占 用 的 系统 资源 〈 如 文件 句柄 ) 
注释 (9D)。 除 非 InputFile 对 象 使 用 完毕 ， 而 且 到 了 需要 弃 之 不 用 的 时 
候 ， 人 否则 不 应 进行 清除 。 大 家 可 能 想 把 这 样 的 机 制 置 入 一 个 finalize() 方 
法 内 ， 但 正如 第 4 章 指出 的 那样 ， 并 非 总 能 保证 finalize0 获 得 正确 的 调用 

(即便 确定 它 会 调用 ， 也 不 知道 何 时 开始 ) 。 这 属于 Java 的 一 项 缺陷 
一 一 除 内 存 清 除 之 外 的 所 有 清除 都 不 会 自动 进行 ， 所 以 必须 知 会 客户 程 
序 员 ， 告 诉 他 们 有 责任 用 finalizeO) 保 证 清除 工作 的 正确 进行 。 


©: 在 C++ 里 , “破坏 器 ?可 帮 我 们 控制 这 一 局 面 。 














在 Cleanup.java 中 ， 我 们 创建 了 一 个 InputFile， 用 它 打 开 用 于 创建 程序 的 
相同 的 源 文 件 。 同 时 一 次 读 取 该 文件 的 一 行内 容 ， 而 且 添加 相应 的 行 
和 





这 个 示例 也 同 大 家 展示 了 为 何在 本 书 的 这 个 地 方 引入 违例 的 概念 。 违 例 
与 Java 的 编程 具有 很 高 的 集成 度 ， 这 主要 是 由 于 编译 器 会 强制 它们 。 只 
有 知道 了 如 何 操作 那些 违例 ， 才 可 更 进一步 地 掌握 编 诺 器 的 知识 。 














9.8 违例 匹配 


“ 掷 ?出 一 个 违例 后 ， 违 例 控制 系统 会 按 当 初 编写 的 顺序 搜索 “最 接近 ”的 
控制 器 。 一 旦 找到 相符 的 控制 器 ， 惑 认为 违例 已 得 到 控制 ， 不 再 进行 更 
多 的 搜索 工作 。 

在 违例 和 它 的 控制 器 之 间 ， 并 不 需要 非常 精确 的 匹配 。 一 个 衍生 类 对 象 
可 与 基础 类 的 一 个 控制 器 相配 ， 如 下 例 所 示 : 


//: Human.java 





// Catching Exception Hierarchies 
class Annoyance extends Exception {} 
class Sneeze extends Annoyance {} 
public class Human { 
public static void main(String[] args) { 
try { 
throw new Sneeze(); 
} catch(Sneeze s) { 
System.out.println("Caught Sneeze"); 
} catch(Annoyance a) { 


System.out.printiln( "Caught Annoyance"); 


} ///:~ 


Sneeze 违 例会 被 相符 的 第 一 个 catch 从 名 捕获 。 当 然 ， 这 只 是 第 一 个 。 然 
而 ， 假 如 我 们 删除 第 一 个 catch 从 句 : 


try { 


throw new Sneeze(); 
} catch(Annoyance a) { 


System.out.println( "Caught Annoyance"); 


那么 剩 下 的 catch 从 名 依然 能 够 工作 ， 因 为 它 捕获 的 是 Sneeze 的 基础 类 。 
换言之 ，catch(Annoyance e) 能 捕获 一 个 Annoyance 以 及 从 它 衍生 的 任何 
类 。 这 一 点 非常 重要 ， 因 为 一 旦 我 们 决定 为 一 个 方法 添加 更 多 的 违例 ， 
而 且 它 们 都 是 从 相同 的 基础 类 继承 的 ， 那 么 客户 程序 员 的 代码 就 不 需要 
更 改 。 至 少 能 够 假定 它们 捕获 的 是 基础 类 。 


一 位 ， 试 图 “屏蔽 ”衍生 类 违例 ， 就 象 下 面 这 








try { 


throw new Sneeze(); 
} catch(Annoyance a) { 
System.out.println( "Caught Annoyance"); 
} catch(Sneeze s) { 


System.out.println("Caught Sneeze"); 


则 编译 器 会 产生 一 条 出 错 消息 ， 因 为 它 发 现 永远 不 可 能 抵达 Sneeze 捕 获 
从 名 o 


9.8.1 违例 准则 

用 违例 做 下 面 这 些 事情 : 

(1) 解决 问题 并 再 次 调用 造成 违例 的 方法 。 

(2) 平息 事态 的 发 展 ， 并 在 不 重新 尝试 方法 的 前 提 下 继续 。 
(3) 计算 另 一 些 结果 ， 而 不 是 希望 方法 产生 的 结果 。 


(4) 在 当前 环境 中 尽 可 能 解决 问题 ， 以 及 将 相同 的 违例 重新 “ 掷 ? 出 一 个 
更 高 级 的 环境 。 


(5) 在 当前 环境 中 尽 可 能 解决 问题 ， 以 及 将 不 同 的 违例 重新 “ 掷 ? 出 一 个 
更 高 级 的 环境 。 


(6) 中 止 程序 执行 。 
(7) ”简化 编码 。 寿 违例 方案 使 事情 变 得 更 加 复杂 ， 那 整 会 令 人 非常 烦 
恼 ， 不 如 不 用 。 


(8) 使 自己 的 库 和 程序 变 得 更 加 安全 。 这 既是 一 种 “短期 投资 "《〈 便 于 调 
试 ) ， 也 是 一 种 “长 期 投资 * (改善 应 用 程序 的 健壮 性 ) 














9.9 忌 结 


通过 先进 的 错误 纠正 与 恢复 机 制 ， 我 们 可 以 有 效 地 增强 代码 的 健壮 程 
度 。 对 我 们 编写 的 每 个 程序 来 说 ， 错 误 恢 复 都 属于 一 个 基本 的 考虑 目 
标 。 它 在 Java 中 显得 尤为 重要 ， 因 为 该 语言 的 一 个 目标 就 是 创建 不 同 的 
程序 组 件 ， 以 便 其 他 用 户 《 客 户 程 序 员 ) 使 用 。 为 构建 一 套 健壮 的 系 
统 ， 每 个 组 件 都 必须 非常 健壮 。 


在 Java 里 ， 违 例 控制 的 目的 是 使 用 尽 可 能 精简 的 代码 创建 大 型 、 可 靠 的 
应 用 程序 ， 同 时 排除 程序 里 那些 不 能 控制 的 错误 。 


违例 的 概念 很 难 掌握 。 但 只 有 很 好 地 运用 它 ， 才 可 使 自己 的 项 目 立 即 获 
得 显著 的 收益 。Java 强 迫 遵 守 违 全 所 有 方面 的 问题 ， 所 以 无 论 库 设计 者 
还 是 客户 程序 员 ， 都 能 够 连续 一 致 地 使 用 它 。 








9.10 练习 


(1) 用 main0 创 建 一 个 类 ， 令 其 搬出 try 块 内 的 Exception 类 的 一 个 对 象 。 
为 Exception 的 构建 器 赋予 一 个 字 串 参数 。 在 catch 从 句 内 捕获 违例 ， 并 打 
de aoe HA finally AE, FFT EU —- A, EH A GAIE 
到 达 那 里 。 


(2) 用 extends 关 键 字 创建 自己 的 违例 类 。 为 这 个 类 写 一 个 构建 器 ， 令 其 
采用 String 参 数 ， 并 随同 String 句 柄 把 它 保存 到 对 象 内 。 写 一 个 方法 ， 令 
其 打印 出 保存 下 来 的 String。 创 建 一 个 try-catch 从 多， 练习 实际 操作 新 违 
例 。 


(3) 写 一 个 类 ， 并 令 一 个 方法 掷 出 在 练习 2 中 创建 的 类 型 的 一 个 违例 。 试 
着 在 没有 违例 规范 的 前 提 下 编译 它 ， 观 察 编译 器 会 报告 什么 。 接 着 添加 
适当 的 违例 规范 。 在 一 个 try-catch 从 句 中 党 试 自己 的 类 以 及 它 的 违例 。 


(4) 在 第 5 章 ， 找 到 调用 了 Assert,java 的 两 个 程序 ， 并 修改 它们 ， 令 其 掷 
出 自己 的 违例 类 型 ， 而 不 是 打印 到 System.err。 该 违例 应 是 扩展 了 


RuntimeException 的 一 个 内 部 类 。 


wr. 
10 Java I0 
-语言 设计 人 员 来 说 ， 创 建 好 的 给 入 / 输 由 系统 是 一 项 特别 困难 的 人 








由 于 存在 大 量 不同 的 设计 方案 ， 所 以 该 任务 的 困难 性 是 很 容易 证 明 的 。 
其 中 最 大 的 挑战 似乎 是 如 何 履 盖 所 有 可 能 的 因素 。 不 仅 有 三 种 不 同 的 种 
类 的 IO 需 要 考虑 (文件 、 控 制 台 、 网 络 连 接 ) ， 而 且 需 要 通过 大 量 不 同 
的 廊 式 写 它 们 通信 ME. BIW En SES BRATS BESS 
aia 


Java 库 的 设计 者 通过 创建 大 量 类 来 攻克 这 个 难题 。 事 实 上 ，Java 的 IO 系 
统 采用 了 如 此 多 的 类 ， 以 致 刚 开 始 会 产生 不 知 从 何 处 入 手 的 感 党 〈 有 具有 
讽刺 意味 的 是 ，Java 的 IO 设计 初衷 实际 要 求 避免 过 多 的 类 ) 。 从 Java 1.0 
升级 到 Java” 1.1 后，IO 库 的 设计 也 发 生 了 显 背 的 变化 。 此 时 并 非 简单 地 
用 新 库 蔡 换 旧 库 ，Sun 的 设计 人 员 对 原来 的 库 进 行 了 大 手笔 的 扩展 ， 添 
加 了 大 量 新 的 内 容 。 因 此 ， 我 们 有 时 不 得 不 混合 使 用 新 库 与 旧 库 ， 产 生 
令 人 无 条 的 复杂 代码 。 


本 章 将 帮助 大 家 理解 标准 Java 库 内 的 各 种 IO 类 ， 并 学 习 如 何 使 用 它们 。 
本 章 的 第 一 部 分 将 介绍 “ 旧 ” 的 Java 1.0 IO 流 库 ， 因 为 现在 有 大 量 代 码 仍 
在 使 用 那个 库 。 本 章 剩 下 的 部 分 将 为 大 家 引入 Java 1.1 IO 库 的 一 些 新 特 
性 。 注 意 奉 用 Java ” 1.1 编译 器 来 编译 本 章 第 一 部 分 介绍 的 部 分 代码 ， 可 
能 会 得 到 一 条 “不 建议 使 用 该 特性 ”(Deprecated feature) 警告 消息 。 代 
码 仍然 能 够 使 用 ;编译 器 只 是 建议 我 们 换 用 本 章 后 面 要 讲述 的 一 些 新 特 
性 。 但 我 们 这 样 做 是 有 价值 的 ， 因 为 可 以 更 清楚 地 认识 老 方法 与 新 方法 
ee eens cary 
RAZ) 

















10.1 输入 和 输出 


可 将 Java 库 的 IO 类 分 割 为 输入 与 输出 两 个 部 分 ， 这 一 点 在 用 Web 浏 览 器 
阅读 联机 Java 类 文档 时 便 可 知道 。 通 过 继承 ， 从 InputStream 〈 输 入 流 ) 
衍生 的 所 有 类 都 拥有 名 为 read0) 的 基本 方法 ， 用 于 读 取 和 单个 字 节 或 者 字 
节 数 组 。 类 似 地 ， 从 OutputStream 簿 生 的 所 有 类 都 拥有 基本 方法 
write()， 用 于 写 入 单个 字 市 或 者 字 节 数组 。 人 然而， 我 们 通常 不 会 用 到 这 
些 方法 ; 它们 之 所 以 存在 ， 是 因为 更 复杂 的 类 可 以 利用 它们 ， 以 便 提 供 
一 个 更 有 用 的 接口 。 因 此 ， 我 们 很 少 用 单个 类 创建 自己 的 系统 对 象 。 一 
般 情 况 下 ， 我 们 都 是 将 多 个 对 象 重 登 在 一 起 ， 提 供 上 自己 期 望 的 功能 。 我 
们 之 所 以 感到 Java 的 流 库 (Stream Library) 异常 复杂 ， 正 是 由 于 为 了 创 
建 单独 一 个 结果 流 ， 却 需要 创建 多 个 对 象 的 缘故 。 


很 有 必要 按照 功能 对 类 进行 分 类 。 库 的 设计 者 首先 决定 与 输入 有 关 的 所 
有 类 都 从 ImputStream 继 承 ， 而 与 输出 有 关 的 所 有 类 都 从 OutputStream 继 
TKK 

10.1.1 InputStream 的 类 型 


InputStream 的 作用 是 标志 那些 从 不 同 起 源 地 产生 输入 的 类 。 这 些 起 源 地 
包括 (每 个 都 有 一 个 相关 的 InputStream 子 类 ) : 


(1) 字 节 数组 


(2) String 对 象 

















(3) 文件 


( “管道 *， 它 的 工作 原理 与 现实 生活 中 的 管道 类 似 : 将 一 些 东西 置 入 
一 端 ， 它 们 在 另 一 端 出 来 。 (5) 一 系列 其 他 流 ， 以 便 我 们 将 其 统一 收集 
到 单独 一 个 流 内 。 


(6) 其 他 起 源 地 ， 如 Internet 连 接 等 〈 将 在 本 书后 面 的 部 分 讲述 ) 。 
除 此 以 外 ，FilterInputStream 也 属于 InputStream 的 一 种 类 型 ， 用 它 可 


为 “破坏 器 ”类 提供 一 个 基础 类 ， 以 便 将 属性 或 者 有 用 的 接口 同 输入 流连 
接 到 一 起 。 这 将 在 以 后 讨论 。 


Constructor 


Class Function 
Arguments 


Allows a buffer inThe buffer from which 
memory to be used|to extract the bytes. 


As a source of data. Connect it 
to a FilterInputStream object 
to provide a useful interface. 


StringBuffer-InputStream {Converts a String/A String. The 


anjunderlying 
implementation actually 
uses a StringBuffer. 


As a source of data. Connect it 
to a FilterInputStream object 
to provide a useful interface. 


File-InputStream 


As a source of data. Connect it 
to a FilterInputStream object 
to provide a useful interface. 





类 功能 构建 器 参数 / 如 何 使 用 


ByteArrayInputStream 人 允许 内 存 中 的 一 个 绥 冲 区 作为 InputStream 使 用 从 
中 提取 字 市 的 缓冲 区 / 作为 一 个 数据 源 使 用 。 通 过 将 其 同一 个 
FilterInputStream 对 象 连接 ， 可 提供 一 个 有 用 的 接口 


StringBufferInputStream 将 一 个 String 转 换 成 InputStream 一 个 String 〈 字 
P) 。 基 础 的 实施 方案 实际 来 用 一 个 StringBuffer( 字 上 串 绥 冲 ) / 作为 一 
个 数据 源 使 用 。 通 过 将 其 同一 个 FilterInputStream 对 象 连接 ， 可 提供 一 个 
有 用 的 接口 


FileInputStream 用 于 从 文件 读 取 信息 代表 文件 名 的 一 个 String， 或 者 一 
个 File 或 FileDescriptor 对 象 / 作为 一 个 数据 源 使 用 。 通 过 将 其 同一 个 
FilterInputStream 对 象 连接 ， 可 提供 一 个 有 用 的 接口 


Piped-InputStream {Produces the data 
being written to 
associated 

Implements 
“piping” concept. 


As a source of data in 
multithreading. Connect 
it to a 
FilterInputStream 
object to provide a 
useful interface. 


Coverts two 
or an 


single InputStream. 





As a source of data. 
Connect it to a 


object to provide a 
useful interface. 





Abstract class which is anj/See Table 10-3. 
interface for decorators that 
provide useful functionality 


to the other InputStream 
classes. See Table 10-3. 


See Table 10-3. 





PipedInputString ”产生 为 相关 的 PipedOutputStream 写 的 数据 。 实 现 了 “ 管 
道 化 ”的 概念 ”PipedOutputStream / 作为 一 个 数据 源 使 用 。 通 过 将 其 同一 
个 FilterInputStream 对 象 连接 ， 可 提供 一 个 有 用 的 接口 


SequenceInputStream 将 两 个 或 更 多 的 InputStream 对 象 转换 成 单个 
InputStream 使 用 两 个 InputStream 对 象 或 者 一 个 Enumeration， 用 于 
InputStream 对 象 的 一 个 容器 / 作为 一 个 数据 源 使 用 。 通 过 将 其 同一 个 
FilterInputStream 对 象 连接 ， 可 提供 一 个 有 用 的 接口 


FilterInputStream “对 作为 破坏 器 接口 使 用 的 类 进行 抽象 ， 那 个 破坏 器 为 
其 他 InputStream 类 提供 了 有 用 的 功能 。 参 见 表 10.3 参见 表 10.3 / 参见 表 
10.3 

10.1.2 OutputStream 的 类 型 

这 一 类 别 包括 的 类 决定 了 我 们 的 输入 往 何 处 去 : 一 个 字 节 数组 (但 没有 
inch 假定 我 们 可 用 字 节 数组 创建 一 个 ); 一 个 文件 ; 或 者 一 个 “ 管 
Je” o 


BRIELASS, FilterOutputStream 为 “破坏 器 ?类 提供 了 一 个 基础 类 ， 它 将 属 
性 或 者 有 用 的 接口 同 输出 流连 接 起 来 。 这 将 在 以 后 讨论 。 


表 10.2 OutputStream 的 类 型 


| | | 





Class Function Constructor 
Arguments 


neat. 


ByteArray- 
OutputStream 





To designate the 
destination of your data. 
Connect it to a 
FilterOutputStream 
object to provide a useful 
interface. 


File-OutputStream 


To designate the 
destination of your data. 
Connect it to a 
FilterOutputStream 
object to provide a useful 
interface. 


Piped-OutputStream 


Creates a buffer inloptional initial size 
memory. All the data thatllof the buffer. 

you send to the stream is 

placed in this buffer. 


For sending information tolA String 

a file. representing the 
file name, or a File 
or FileDescriptor 
object. 


Any information you write|PipedInputStream 
to this automatically ends 
as input for the 
PipedInput- 
Implements the 
“piping” concept. 


To designate the 
destination of your data for 
multithreading. Connect it 
to a FilterOutputStream 
object to provide a useful 
interface. 





Filter-OutputStream Abstract class which is anllSee Table 10-4. 
for decorators 
provide useful 
functionality to the other 


OutputStream classes. 
See Table 


10-4. 


See Table 10-4. 





类 功能 构建 器 参数 / 如 何 使 用 


ByteArrayOutputStream 在 内 存 中 创建 一 个 缓冲 区 。 我 们 发 送 给 流 的 所 有 
数据 都 会 置 入 这 个 缓冲 区 。 “可 选 缓冲 区 的 初始 大 小 / 用 于 指出 数据 的 
目的 地 。 若 将 其 同 FilterOutputStream 对 象 连接 到 一 起 ， 可 提供 一 个 有 用 
的 接口 


FileOutputStream 将 信息 发 给 一 个 文件 用 一 个 String 代 表 文 件 名 ， 或 选用 
一 个 File 或 FileDescriptor 对 象 / 用 于 指出 数据 的 目的 地 。 知 将 其 同 
FilterOutputStream 对 象 连接 到 一 起 ， 可 提供 一 个 有 用 的 接口 


PipedOutputStream 我 们 写 给 它 的 任何 信息 都 会 自动 成 为 相关 的 
PipedInputStream 的 输出 。 实 现 了 “管道 化 ”的 概念 ”PipedInputStream / 为 
多 线程 处 理 指出 自己 数据 的 目的 地 / 将 其 同 FilterOutputStream 对 象 连接 
到 一 起 ， 便 可 提供 一 个 有 用 的 接口 


FilterOutputStream 对 作为 破坏 器 接口 使 用 的 类 进行 抽象 处 理 ， 那 个 破坏 











器 为 其 他 OutputStream 类 提供 了 有 用 的 功能 。 参 见 表 10.4 参见 表 10.4 / 
参见 表 10.4 


10.2 增添 属性 和 有 用 的 接口 


利用 层次 化 对 象 动态 和 透明 地 添加 单个 对 象 的 能 力 的 做 法 叫 作 “装饰 

器 ”(Decorator ) 方案 “方案 ”属于 本 书 第 16 章 的 主题 GERD) 。 
装饰 器 方案 规定 封装 于 初始 化 对 象 中 的 所 有 对 象 都 拥有 相同 的 接口 ， 以 
便利 用 装饰 器 的 “透明 ”性 质 我 们 将 相同 的 消息 发 给 一 个 对 象 ， 无 论 
它 是 否 已 被 “装饰 ?。 这 正 是 在 Java IO 库 里 存在 “过 滤器 ” (Filter) 类 的 原 
Al: 抽象 的 “过 滤器 ”类 是 所 有 装饰 器 的 基础 类 【装饰 器 必须 拥有 与 它 装 
饰 的 那个 对 象 相同 的 接口 ， 但 装饰 右 亦 可 对 接口 作出 扩展 ， 这 种 情况 见 
诸 于 几 个 特殊 的 “过 小 器 ”类 中 ) 。 


子 类 处 理 要 求 大 量子 类 对 每 种 可 能 的 组 合 提供 支持 时 ， 便 经 第 会 用 到 装 
饰 器 一 一 由 于 组 合 形式 太 多 ， 造 成 子 类 处 理 变 得 不 切实 际 。Java IO 库 要 
求 许多 不 同 的 特性 组 合 方案 ， 这 正 是 装饰 器 方案 显得 特别 有 用 的 原因 。 
但 是 ， 装 饰 器 方案 也 有 自己 的 一 个 缺点 。 在 我 们 写 一 个 程序 的 时 候 ， 装 
饰 占 为 我 们 提供 了 大 得 多 的 灵活 性 (因为 可 以 方便 地 混合 与 匹配 属 
性 ) ， 但 它们 也 使 自己 的 代码 变 得 更 加 复杂 。 原 因 在 于 Java IO 库 操作 不 
便 ， 我 们 必须 创建 许多 类 一 一 “核心 "IO 类 型 加 上 所 有 装饰 器 一 一 才能 得 
到 目 己 希 望 的 单个 IO 对 象 。 


FilterInputStream 和 FilterOutputStream 〈 这 两 个 名 字 不 十 分 直观 ) 提供 了 
相应 的 装饰 器 接口 ， 用 于 控制 一 个 特定 的 输入 流 〈InputStream) 或 者 输 
出 流 (OutputStream) 。 它 们 分 别 是 从 InputStream 和 OutputStream 衍 生出 
来 的 。 此 外 ， 它 们 都 属于 抽象 类 ， 在 理论 上 为 我 们 与 一 个 流 的 不 同 通信 
手段 都 提供 了 一 个 通用 的 接口 。 事 实 上 ，FilterInputStream 和 和 
FilterOutputStream 只 是 简单 地 模仿 了 自己 的 基础 类 ， 它 们 是 一 个 装饰 器 
的 基本 要 求 。 


10.2.1 通过 FilterInputStream 从 InputStream 里 读 入 数据 


FilterInputStream 类 要 完成 两 件 全 然 不 同 的 事情 。 其 中 ，DataInputStream 
允许 我 们 读 取 不 同 的 基本 类 型 数据 以 及 String 对 象 〈 所 有 方法 都 

以 “read” 开 头 ， 比 如 readByte0，readFloat0 等 等 ) 。 伴 随 对 应 的 
DataOutputStream， 我 们 可 通过 数据 “ 流 ” 将 基本 类 型 的 数据 从 一 个 地 方 
搬 到 另 一 个 地 方 。 这 些 * 地 方 ” 是 由 表 10.1 总 结 的 那些 类 决定 的 。 知 读 取 
块 内 的 数据 ， 并 自己 进行 解析 ， 束 不 需要 用 到 DataInputStream。 但 在 其 

















他 许多 情况 下 ， 我 们 一 般 都 想 用 它 对 自己 读 入 的 数据 进行 自动 格式 化 。 


剩 下 的 类 用 于 修改 InputStream 的 内 部 行为 方式 : 是 否 进行 缓冲 ， 和 是 否 跟 
踩 上 自己 读 入 的 数据 行 ， 以 及 是 否 能 够 推 回 一 个 字符 等 等 。 后 两 种 类 看 起 
来 特别 象 提 供 对 构建 一 个 编译 占 的 支持 换言之， 添加 它们 为 了 文 持 
Java 编 译 器 的 构建 ) ， 所 以 在 种 规 编程 中 一 般 都 用 不 着 它们 。 


也 许 儿 乎 每 次 都 要 缓冲 目 己 的 输入 ， 无 论 连 接 的 是 哪个 IO 设备 。 所 以 IO 
库 最 明智 的 做 法 就 是 将 未 缓冲 输入 作为 一 种 特殊 情况 处 理 ， 同 时 将 缓冲 
和 输入 接纳 为 标准 做 法 。 


表 10.3 FilterInputStream 的 类 型 
Class Function Constructor 
Arguments 


Data-InputStream ||Used in concert withInputStream 
DataOutputStream, so you can read 
primitives (int, char, long, etc.) from a 
stream in a portable fashion. 





to 
primitive types. 


Buffered-InputStream this to prevent aj/InputStream, 


want more data.lilbuffer size. 
saying “Use 





This doesn’t provide an 


interface per se, just a 
requirement that a buffer be 
used. Attach an interface object. 


LineNumber-InputStream track of  linelInputStream 
in the input 
; you can call 
getLineNumber( ) and 
setLineNumber(int). 


This just adds line numbering, 
so you'll probably attach an 
interface object. 


Pushback-InputStream 
buffer so that you can push 
back the last character read. 


Generally used in the scanner 
for a compiler and probably 
included because the 
compiler needed it. 
probably won’t use this. 





类 功能 构建 器 参数 / 如 何 使 用 


DatalnputStream 与 DataOutputStream 联 合 使 用 ， 使 自己 能 以 机 动 方式 读 
取 一 个 流 中 的 基本 数据 类 型 (int，char，long 等 等 ) InputStream/ 包 含 了 
一 个 完整 的 接口 ， 以 便 读 取 基 本 数据 类 型 


BufferedInputStream 避免 每 次 想 要 更 多 数据 时 都 进行 物理 性 的 谈 取 ， 告 
诉 它 “请 先 在 缓冲 区 里 找 ”InputStream， 没 有 可 选 的 缓冲 区 大 小 / 本 身 并 
不 能 提供 一 个 接口 ， 只 是 发 出 使 用 缓冲 区 的 要 求 。 要 求 同 一 个 接口 对 象 
连接 到 一 起 


LineNumberInputStream ”跟踪 输入 流 中 的 行 号 ; 可 调用 getLineNumber() 





以 及 setLineNumber(int) 只 是 添加 对 数据 行 编号 的 能 力 ， 所 以 可 能 需要 同 
一 个 真正 的 接口 对 象 连接 


PushbackInputStream 有 一 个 字 节 的 后 推 缓冲 区 ， 以 便 后 推 读 入 的 上 一 个 
字符 InputStream / 通常 由 编译 占 在 扫 插 右 中 使 用 ， 因 为 Java 编 译 右 需要 
它 。 一 般 不 在 自己 的 代码 中 使 用 


10.2.2 通过 FilterOutputStream 问 OutputStream 里 写 入 数据 


与 DataInputStream 对 应 的 是 DataOutputStream， 后 者 对 各 个 基本 数据 类 
型 以 及 String 对 象 进行 格式 化 ， 并 将 其 置 入 一 个 数据 “ 流 ” 中 ， 以 便 任何 
机 器 上 的 DataInputStream 都 能 正常 地 读 取 它们 。 所 有 方法 都 以 “wirte”* 开 
头 ， 例 如 writeByte0，writeFloat0 等 等 。 


若 想 进行 一 些 真正 的 格式 化 输出 ， 比 如 输出 到 控制 台 ， 请 使 用 
PrintStream。 利 用 它 可 以 打印 出 所 有 基本 数据 类 型 以 及 String 对 象 ， 并 
可 采用 一 种 易于 查看 的 格式 。 这 与 DataOutputStream 正 好 相反 ， 后 者 的 
目标 是 将 那些 数据 置 入 一 个 数据 流 中 ， 以 便 DataInputStream 能 够 方便 地 
重新 构造 它们 。System.out 静 态 对 象 是 一 个 PrintStream。 


PrintStream 内 两 个 重要 的 方法 是 print() 和 println()。 它 们 已 进行 了 和 窗 六 处 
理 ， 可 打印 出 所 有 数据 闫 型 。printO0 和 printin0 之 间 的 差异 是 后 者 在 操作 
完毕 后 会 自动 添加 一 个 新 行 。 
BufferedOutputStream 属 于 一 种 “修改 器 "， 用 于 指示 数据 流 使 用 缓冲 技 
术 ， 使 自己 不 必 每 次 都 向 流 内 物理 性 地 写 入 数据 。 通 常 都 应 将 它 应 用 于 
文件 处 理 和 控制 占 IO。 


表 10.4 FilterOutputStream 的 类 型 











nstr r 
Class Function Constructo 
Arguments 


Data-OutputStream liUsed in concert withlOutputStream 
DataInputStream so you can 


write primitives (int, char, 
long, etc.) to a stream in a 
portable fashion. 










Contains full interface 
to allow you to write 


PrintStream 


Should be the “final” 
wrapping for your 
OutputStream object. 





Use this to prevent a physicaljOutputStream, 
write every time you send ajjwith optional buffer 
piece of data. You’re sayingllsize. 
“Use a buffer.” You can call 
flush( ) to flush the buffer. 


This doesn’t provide 


an interface per se, just 
a requirement that a 
buffer is used. Attach 
an interface object. 


类 功能 构建 器 参数 / 如 何 使 用 


DataOutputStream 与 DataInputStream 配 合 使 用 ， 以 便 采 用 方便 的 形式 将 
基本 数据 类 型 Cint, char, long) 写 入 一 个 数据 流 OutputStream / 包 
含 了 完整 接口 ， 以 便 我 们 写 入 基本 数据 类 型 


PrintStream 用 于 产生 格式 化 输出 。DataOutputStream 控 制 的 是 数据 的 “ 存 
fig”, 而 PrintStream 控 制 的 是 “显示 ”OutputStream， 可 选 一 个 布尔 参数 ， 
指示 绥 冲 区 是 否 与 每 个 新 行 一 同 刷 新 / 对 于 目 己 的 OutputStream 对 象 ， 
应 该 用 “final* 将 其 封闭 在 内 。 可 能 经 常 都 要 用 到 它 


BufferedOutputStream 用 它 避 免 每 次 发 出 数据 的 时 候 都 要 进行 物理 性 的 
写 入 ， 要 求 它 “请 先 在 缓冲 区 里 找 ”。 可 调用 ftush0， 对 缓冲 区 进行 刷新 
OutputStream， 可 选 缓冲 区 大 小 /本 号 并 不 能 提供 一 个 接口 ， 只 是 发 出 
使 用 缓冲 区 的 要 求 。 需 要 同一 个 接口 对 象 连接 到 一 起 








10.3 ASA: RandomAccessFile 


RandomAccessFile 用 于 包含 了 已 知 长 度 记 录 的 文件 ， 以 便 我 们 能 
seek() 从 一 条 记录 移 人 至 男 一 条 ; 然后 读 取 或 修改 那些 记录 。 各 记录 的 长 
度 并 不 一 定 相 同 ， 只 要 知道 它们 有 多 大 以 及 置 于 文件 何 处 即 可 。 


首先 ， 我 们 有 点 难以 相信 RandomAccessFile 不 属于 InputStream 或 者 
OutputStream 分 层 结构 的 一 部 分 。 除 了 恰巧 实现 了 DataInput 以 及 
DataOutput 〈 这 两 者 亦 由 DataInputStream 和 DataOutputStream 实 现 ) 接口 
之 外 ， 它 们 与 那些 分 层 结构 并 无 什么 关系 。 它 甚至 没有 用 到 现 有 
InputStream 或 OutputStream 类 的 功能 一 一 采用 的 是 一 个 完全 不 相干 的 
类 。 该 类 属于 全 新 的 设计 ， 含 有 自己 的 全 部 (大 多 数 为 固有 ) FE 之 
所 以 要 这 样 做 ， 是 因为 RandomAccessFile 拥 有 与 其 他 IO 类 型 完全 不 同 的 
行为 ， 因 为 我 们 可 在 一 个 文件 里 向 前 或 同 后 移动 。 不 管 在 哪 种 情况 下 ， 
它 都 是 独立 运作 的 ， 作 为 Object 的 一 个 “直接 继承 人 ”使 用 。 


从 根本 上 说 ，RandomAccessFile 类 似 DataInputStream 和 DataOutputStream 
的 联合 使 用 。 其 中 ，getFilePointerO0 用 于 了 解 当 前 在 文件 的 什么 地 方 ， 
seek0 用 于 移 至 文件 内 的 一 个 新 地 点 ， 而 lengthO0 用 于 判断 文件 的 最 大 长 
度 。 此 外 ， 构 建 器 要 求 使 用 另 一 个 自 变 量 〈( 与 C 的 fopen0O) 完 全 一 样 )， 
指出 自己 只 是 随机 读 ("r") ， 还 是 读 写 兼 施 Crw") 。 这 里 没有 提供 
对 “只 写 文件 ”的 文 持 。 也 就 是 说 ， 假 如 是 从 DataInputStream 继 承 的 ， 那 
么 RandomAccessFile 也 有 可 能 能 很 好 地 工作 。 


还 有 更 难 对 付 的 。 很 容易 想象 我 们 有 时 要 在 其 他 类 型 的 数据 流 中 搜索 ， 
比如 一 个 ByteArrayInputStream， 但 搜索 方法 只 有 RandomAccessFile 才 会 
提供 。 而 后 者 只 能 针对 文件 才能 操作 ， 不 能 针对 数据 流 操 作 。 此 时 ， 
BufferedInputStream 确 实 允 许 我 们 标记 一 个 位 置 ( 使 用 mark()， 它 的 值 
容纳 于 单个 内 部 变量 中 ) ， 并 用 reset0 重 设 那 个 位 置 。 但 这 些 做 法 都 存 
在 限制 ， 并 不 是 特别 有 用 。 

















10.4 File 类 


File 类 有 一 个 欺骗 性 的 名 字 一 一 通常 会 认为 它 对 付 的 是 一 个 文件 ， 但 实 
情 并 非 如 此 。 它 既 代 表 一 个 特定 文件 的 名 字 ， 也 代表 目录 内 一 系列 文件 
的 名 字 。 知 代表 一 个 文件 集 ， 便 可 用 list(0) 方 法 查询 这 个 集 ， 返 回 的 是 一 

个 字 串 数组 。 之 所 以 要 返回 一 个 数组 ， 而 非 某 个 灵活 的 集合 类 ， 是 因为 
元 系 的 数量 是 固定 的 。 而 且 知 想得到 一 个 不 同 的 目录 列表 ， 只 需 创 建 一 
个 不 同 的 File 对 象 即 可 。 事 实 上 ,，“FilePath” (文件 路 径 ) 似乎 是 一 个 更 
好 的 名 字 。 本 节 将 同 大 家 完整 地 例 示 如 何 使 用 这 个 类 ， 其 中 包括 相关 的 
FilenameFilter (文件 名 过 滤器 ) 接口 。 


10.4.1 目录 列表 器 


现在 假设 我 们 想 观 看 一 个 目录 列表 。 可 用 两 种 方式 列 出 File 对 象 。 知 在 
不 含 自 变量 (SEO 的 情况 下 调用 ]ist()， 会 获得 File 对 象 包含 的 一 个 完 
整 列表 。 然 而 ， 知 想 对 这 个 列表 进行 某 些 限制 ， 就 需要 使 用 一 个 “目录 
过 滤器 ”， 该 类 的 作用 是 指出 应 如 何 选择 File 对 象 来 完成 显示 。 


下 面 是 用 于 这 个 例子 的 代码 或 在 执行 该 程序 时 过 到 困难 ， 请 参考 第 3 
章 3.1.2 小 节 “ 赋 值 ?) : 


























//: DirList.java 


// Displays directory listing 
package c10; 
import java.io.*; 
public class DirList { 
public static void main(String[] args) { 
try { 
File path = new File("."); 


String[] list; 


if(args.length == 0) 
list = path.list(); 
else 
list = path.list(new DirFilter(args[0])); 
for(int i = 0; i < list.length; i++) 
System.out.println(list[i]); 
} catch(Exception e) { 


e.printStackTrace(); 


J 


class DirFilter implements FilenameFilter { 
String afn; 
DirFilter(String afn) { this.afn = afn; } 
public boolean accept(File dir, String name) { 
// Strip path information: 
String f = new File(name).getName(); 
return f.indexOf(afn) != -1; 


} 
LV] i 


DirFilter 类 “实现 ”J interface FilenameFilter 〈 关 于 接口 的 问题 ， 已 在 第 7 
章 进 行 了 详 述 ) 。 下 面 让 我 们 看 看 FilenameFilter 接 口 有 多 么 简单 : 


public interface FilenameFilter { 





boolean accept( 文 件 目录 , 字 串 名 ); 
} 


它 指出 这 种 类 型 的 所 有 对 象 都 提供 了 一 个 名 为 accept() 的 方法 。 之 所 以 
要 创建 这 样 的 一 个 类 ， 背 后 的 全 部 原因 就 是 把 accept() 方 法 提供 给 list() 方 
法 ， 使 list() 能 够 “回调 *accept()， 从 而 判断 应 将 哪些 文件 名 包括 到 列表 
中 。 因 此 ， 通 常 将 这 种 技术 称 为 “回调 ”"， 有 时 也 称 为 “ 算 子 ”( 也 就 是 
说 ，DirFilter 是 一 个 算 子 ， 因 为 它 唯一 的 作用 就 是 容纳 一 个 方法 ) 。 由 
于 list() 玉 用 一 个 FilenameFilter 对 象 作 为 自己 的 自 变 量 使 用 ， 所 以 我 们 能 
传递 实现 了 FilenameFilter 的 任何 类 的 一 个 对 象 ， 用 它 决定 (甚至 在 运行 
epee 的 行为 方式 。 回 调 的 目的 是 在 代码 的 行为 上 提供 更 大 的 灵 
YE TE o 








通过 DirFilter， 我 们 看 出 尽管 一 个 “接口 ”只 包含 了 一 系列 方法 ， 但 并 不 
局 限于 只 能 写 那 些 方法 〈 但 是 ， 至 少 必 须 提供 一 个 接口 内 所 有 方法 的 定 
义 。 在 这 种 情况 下 ，DirFilter 构 建 器 也 会 创建 ) 。 


accept() 方 法 必须 接纳 一 个 File 对 象 ， 用 它 指示 用 于 寻找 一 个 特定 文件 的 
目录 ; 并 接纳 一 个 String， 其 中 包含 了 要 寻找 之 文件 的 名 字 。 可 决定 使 
用 或 忽略 这 两 个 参数 之 一 ， 但 有 时 至 少 要 使 用 文件 名 。 记 住 list0) 方 法 准 
备 为 目录 对 象 中 的 每 个 文件 名 调用 accept0， 核 实 哪个 应 包含 在 内 一 一 
具体 由 accept0 返 回 的 “布尔 ”结果 决定 。 


为 确定 我 们 操作 的 只 是 文件 名 ， 其 中 没有 包含 路 径 信 息 ， 必 须 采 用 
String 对 象 ， 并 在 它 的 外 部 创建 一 个 File 对 象 。 然 后 调用 getrName()， 它 
的 作用 是 去 除 所 有 路 径 信 息 〈 采 用 与 平台 无 关 的 方式 ) 。 随 后 ， 
accept() 用 String 类 的 indexOf() 方 法 检查 文件 名 内 部 是 否 存 在 搜索 字 

串 "afn"。 硅 在 字 串 内 找到 afn， 那 么 返回 值 就 是 afn 的 起 点 索引 ; 但 假如 
没有 找到 ， 返 回 值 就 是 -1。 注 意 这 只 是 一 个 简单 的 字 串 搜索 例子 ， 未 使 
用 常见 的 表达 式 “ 通 配 符 ” 方 案 ， 比 如 "fo?.b?r*"， 这 种 方案 更 难 实现 。 


list() 方 法 返回 的 是 一 个 数组 。 可 查询 这 个 数组 的 长 度 ， 然 后 在 其 中 遍 
历 ， 选 定数 组 元 素 。 与 C 和 C++ 的 类 似 行为 相 比 ， 这 种 于 方法 内 外 方便 
游历 数组 的 行为 无 疑 是 一 个 显著 的 进步 。 




















1. 匿 名 内 部 类 


下 例 用 一 个 匿名 内 部 类 (已 在 第 7 章 讲 述 ) 来 重 写 显得 非常 理想 。 首 先 
创建 了 一 个 filter0) 方 法 ， 它 返回 指 同 FilenameFilter 的 一 个 句柄 : 


//: DirList2.java 





// Uses Java 1.1 anonymous inner classes 
import java.io.*; 
public class DirList2 { 
public static FilenameFilter 
filter(final String afn) { 
// Creation of anonymous inner class: 
return new FilenameFilter() { 
String fn = afn; 
public boolean accept(File dir, String n) { 
// Strip path information: 
String f = new File(n).getName(); 
return f.indexOf(fn) != -1; 
} 
}; // End of anonymous inner class 
} 
public static void main(String[] args) { 


try { 


File path = new File("."); 


String[] list; 

if(args.length == 0) 
list = path.list(); 

else 
list = path.list(filter(args[0])); 

for(int i = 0; i < list.length; i++) 
System.out.println(list[i]); 

} catch(Exception e) { 


e.printStackTrace(); 


} 
} ///:~ 








注意 filter() 的 目 变 量 必须 是 final。 这 一 点 是 匿名 内 部 类 要 求 的 ， 使 其 能 
使 用 来 自 本 身 作 用 域 以 外 的 一 个 对 象 。 

之 所 以 认为 这 样 做 更 好 ， 是 由 于 FilenameFilter 类 现在 同 DirList2 紧 密 地 
结合 在 一 起 。 然 而 ， 我 们 可 采取 进一步 的 操作 ， 将 匿名 内 部 类 定义 成 
listO 的 一 个 参数 ， 使 其 显得 更 加 精简 。 如 下 所 示 : 


//: DirList3,java 








// Building the anonymous inner class "in-place" 
import java.io.*; 
public class DirList3 { 


public static void main(final String[] args) { 


try { 
File path = new File("."); 
String[] list; 
if(args.length == 0) 
list = path.list(); 
else 
list = path.list( 
new FilenameFilter() { 
public boolean 
accept(File dir, String n) { 
String f = new File(n).getName(); 
return f.indexOf(args[0]) != -1; 
} 
}); 
for(int i = 0; i < list.length; i++) 
System.out.printin(list[i]); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
EFI 














main() 现 在 的 自 变 量 是 final， 因 为 匿名 内 部 类 直接 使 用 args[0]。 


这 展示 了 如 何 利 用 匿名 内 部 类 快速 创建 精简 的 类 ， 以 便 解 决 一 些 复杂 的 
问题 。 由 于 Java 中 的 所 有 东西 都 与 类 有 关 ， 上 所 以 它 无 疑 是 一 种 相当 有 用 
的 编码 技术 。 它 的 一 个 好 处 是 将 特定 的 问题 隔离 在 一 个 地 方 统一 解决 。 
oa 这 样 生 成 的 代码 不 是 十 分 容易 阅读 ， 所 以 使 用 时 必须 慎 


2. 顺序 目录 列表 


经 常 都 需要 文件 名 以 排 好 序 的 方式 提供 。 由 于 Java 1.0 和 Java 1.1 都 没有 
提供 对 排序 的 支持 (从 Java ”1.2 开 始 提 供 ) ， 所 以 必须 用 第 8 章 创 建 的 
SortVector 将 这 一 能 力 直 接 加 入 目 己 的 程序 。 就 象 下 面 这 样 : 


//: SortedDirList.java 


// Displays sorted directory listing 
import java.io.*; 
import c08.*; 
public class SortedDirList { 
private File path; 
private String[] list; 
public SortedDirList(final String afn) { 
path = new File("."); 
if(afn == null) 
list = path.list(); 
else 
list = path.list( 
new FilenameFilter() { 


public boolean 


accept(File dir, String n) { 
String f = new File(n).getName(); 


return f.indexOf(afn) != -1; 


void print() { 
for(int i = 0; i < list.length; i++) 
System.out.printin(list[i]); 
} 
private void sort() { 
StrSortVector sv = new StrSortVector(); 
for(int i = 0; i < list.length; i++) 
sv.addElement(list[i]); 
// The first time an element is pulled from 
// the StrSortVector the list is sorted: 
for(int i = 0; i < list.length; i++) 
list[i] = sv.elementAt(i); 
} 
// Test it: 
public static void main(String[] args) { 


SortedDirList sd; 


if(args.length == 0) 

sd = new SortedDirList(null); 
else 

sd = new SortedDirList(args[0]); 
sd.print(); 


} 
Lif] l= 


这 里 进行 了 另外 少许 改进 。 不 再 是 将 path《〈 路 径 ) Mist GR) 创建 为 
main() 的 本 地 变量 ， 它 们 变 成 了 类 的 成 员 ， 使 它们 的 值 能 在 对 象 “ 生 

存 ” 期 间 方 便 地 访问 。 事 实 上 ，main0 现 在 只 是 对 类 进行 测试 的 一 种 方 
式 。 大 家 可 以 看 到 ， 一 旦 列表 创建 完毕 ， 类 的 构建 器 就 会 自动 开始 对 列 
表 进 行 排 序 。 


这 种 排序 不 要 求 区 分 大 小 写 ， 所 以 最 终 不 会 得 到 一 组 全 部 单词 都 以 大 写 
字母 开 尖 的 列表 ， 跟 痢 是 全 部 以 小 写字 母 开 头 的 列表 。 然 而 ， 我 们 注意 
到 在 以 相同 字母 开头 的 一 组 文件 名 中 ， 大 写字 母 是 排 在 前 面 的 一 一 这 对 
标准 的 排序 来 说 仍 是 一 种 不 合格 的 行为 。Java ”1.2 已 成 功 解决 了 这 个 问 


feo 

















10.4.2 检查 与 创建 目录 


File 类 并 不 仅仅 是 对 现 BREE raf Cl RO JRA 
用 一 个 File 对 象 新 bea 
它 尚 不 存在 的 话 。 亦 可 用 ee [长度 、 ae eee 
读 / 写 属 性 等 ) ， 检 查 一 个 File 对 象 到 底 代 表 一 个 文件 还 是 一 个 目录 ， 
下 列 程序 完整 展示 了 如 何 运 用 File 类 剩 下 的 这 
些 方 > : 




















//: MakeDirectories.java 


// Demonstrates the use of the File class to 


// create directories and manipulate files. 
import java.io.*; 
public class MakeDirectories { 
private final static String usage = 
"Usage:MakeDirectories pathi ...\n" + 
"Creates each path\n" + 
"Usage:MakeDirectories -d pathi ...\n" + 
"Deletes each path\n" + 
"Usage:MakeDirectories -r path1 path2\n" + 
"Renames from path1 to path2\n"; 
private static void usage() { 
System.err.printlin(usage); 
System.exit(1); 
} 
private static void fileData(File f) { 
System.out.printin( 
"Absolute path: " + f.getAbsolutePath() + 
"\n Can read: " + f.canRead() + 
"\n Can write: " + f.canWrite() + 
"\n getName: " + f.getName() + 
"\n getParent: " + f.getParent() + 
"\n getPath: " + f.getPath() + 


"\n length: " + f.length() + 


"\n lastModified: " + f.lastModified()); 
if(f.isFile()) 
System.out.printlin("it's a file"); 
else if(f.isDirectory()) 
System.out.println("it's a directory"); 
} 
public static void main(String[] args) { 
if(args.length < 1) usage(); 
if(args[0].equals("-r")) { 
if(args.length != 3) usage(); 
File 
old = new File(args[1]), 
rname = new File(args[2]); 
old.renameTo(rname) ; 
fileData(old); 
fileData(rname) ; 
return; // Exit main 
} 
int count = 0; 
boolean del = false; 
if(args[0].equals("-d")) { 
count++; 


del = true; 


} 


for( ; count < args.length; count++) { 
File f = new File(args[count]); 
if(f.exists()) { 
System.out.println(f + " exists"); 
if(del) { 
System.out.println("deleting..." + f); 


f.delete(); 


} 


else { // Doesn't exist 
if(!del) { 
f.mkdirs(); 


System.out.println("created " + f); 


} 


fileData(f); 


} ///:~ 





在 fleData() 中 ， 可 看 到 应 用 了 各 种 文件 调查 方法 来 显示 与 文件 或 目录 路 
径 有 关 的 信息 。 


main(0 应 用 的 第 一 个 方法 是 renameTo0， 利 用 它 可 以 重 命名 《或 移动 ) 
一 个 文件 至 一 个 全 新 的 路 径 〈 该 路 径 由 参数 决定 ) ， 它 属于 另 一 个 File 
对 象 。 这 也 适用 于 任何 长 度 的 目录 。 


右 试 验 上 述 程序 ， 就 可 发 现 目 己 能 制作 任意 复杂 程度 的 一 个 目录 路 径 ， 
因为 mkdirs() 会 帮 我 们 完成 所 有 工作 。 在 Java 1.0 中 ，-d 标 志 报告 目录 虽 
然 已 被 删除 ， 但 它 依然 存在 ， 但 在 Java 1.1 中 ， 目 录 会 被 实际 删除 。 

















10.5 IO 流 的 典型 应 用 


尽管 库 内 存在 大 量 IO 流 类 ， 可 通过 多 种 不 同 的 方式 组 合 到 一 起 ， 但 实际 
上 只 有 儿 种 方式 才 会 经 常用 到 。 然 而 ， 必 须 小 心 在 意 才 能 得 到 正确 的 组 
合 。 下 面 这 个 相当 长 的 例子 展示 了 典型 IO 配置 的 创建 与 使 用 ， 可 在 写 目 
己 的 代码 时 将 其 作为 一 个 参考 使 用 。 注 意 每 个 配置 都 以 一 个 注释 形式 的 
编写 起 涉 ， 并 提供 了 适当 的 解释 信息 。 


//: TOStreamDemo. java 

















// Typical IO Stream Configurations 
import java.io.*; 
import com.bruceeckel.tools.*; 
public class I0StreamDemo { 
public static void main(String[] args) { 
try { 
// 1. Buffered input file 
DataInputStream in = 
new DataInputStream( 
new BufferedInputStream( 
new FileInputStream(args[0]))); 
String s, s2 = new String(); 
while((s = in.readLine())!= null) 
s2 += s + "\n"; 
in.close(); 


// 2. Input from memory 


StringBufferInputStream in2 = 
new StringBufferInputStream(s2); 
int c; 
while((c = in2.read()) != -1) 
System.out.print((char)c); 
// 3. Formatted memory input 
try { 
DataInputStream in3 = 
new DataInputStream( 
new StringBufferInputStream(s2)); 
while(true) 
System.out.print((char)in3.readByte()); 
} catch(EOFException e) { 
System.out.printin( 
"End of stream encountered"); 


} 


// 4. Line numbering & file output 
try { 
LineNumberInputStream li = 
new LineNumberInputStream( 
new StringBufferInputStream(s2)); 
DataInputStream in4 = 


new DataInputStream(1i); 


PrintStream out1 = 
new PrintStream( 
new BufferedOutputStream( 
new FileOutputStream( 
"TODemo.out"))); 
while((s = in4.readLine()) != null ) 
out1.printin( 
"Line " + 1i.getLineNumber() + s); 
outi.close(); // finalize() not reliable! 
} catch(EOFException e) { 
System.out.printin( 
"End of stream encountered"); 


} 


// 5. Storing & recovering data 
try { 
DataOutputStream out2 = 
new DataOutputStream( 
new BufferedOutputStream( 
new FileOutputStream("Data.txt"))); 
out2.writeBytes( 
"Here's the value of pi: \n"); 
out2.writeDouble(3.14159); 


out2.close(); 


DataInputStream in5 = 
new DataInputStream( 
new BufferedInputStream( 
new FileInputStream("Data.txt"))); 
System.out.println(in5.readLine()); 
System.out.println(in5.readDouble()); 
} catch(EOFException e) { 
System. out.printin( 
"End of stream encountered"); 
} 
// 6. Reading/writing random access files 
RandomAccessFile rf = 
new RandomAccessFile("rtest.dat", "rw"); 
for(int i = 07 i < 10; i++) 
rf.writeDouble(i*1.414); 
rf.close(); 
rf = 
new RandomAccessFile("rtest.dat", "rw"); 
rf.seek(5*8); 
rf.writeDouble(47.0001); 
rf.close(); 
rf = 


new RandomAccessFile("rtest.dat", "r"); 


for(int i = 0; i < 10; i++) 

System. out.printin( 

"Value "+i4+ ": "+ 
rf.readDouble()); 

rf.close(); 
// 7. File input shorthand 
InFile in6 = new InFile(args[0]); 
String s3 = new String(); 
System.out.printin( 

"First line in file: " + 

in6.readLine()); 

in6.close(); 
// 8. Formatted file output shorthand 
PrintFile out3 = new PrintFile("Data2.txt"); 
out3.print("Test of PrintFile"); 
out3.close(); 
// 9. Data file output shorthand 
OutFile out4 = new OutFile("Data3.txt"); 
out4.writeBytes("Test of outDataFile\n\r"); 
out4.writeChars("Test of outDataFile\n\r"); 
out4.close(); 
catch(FileNotFoundException e) { 


System.out.printin( 


"File Not Found:" + args[0]); 
} catch(IOException e) { 


System.out.printin( "IO Exception"); 


} 
} ///:~ 


10.5.1 输入 流 


当然 ， 我 们 经 常 想 做 的 一 件 事情 是 将 格式 化 的 输出 打印 到 控制 台 ， 但 那 
己 在 第 5 童 创建 的 com.bruceeckel.tools 中 得 到 了 简化 。 


第 1 到 第 4 部 分 演示 了 输入 流 的 创建 与 使 用 《尽管 第 4 部 分 展示 了 将 输出 
流 作为 一 个 测试 工具 的 简单 应 用 ) 。 


1. 缓冲 的 输入 文件 


为 打开 一 个 文件 以 便 输 入 ， 需 要 使 用 一 个 FileInputStream， 同 时 将 一 个 
String 或 File 对 象 作为 文件 名 使 用 。 为 提高 速度 ， 最 好 先 对 文件 进行 缓冲 
处 理 ， 从 而 获得 用 于 一 个 BufferedInputStream 的 构建 器 的 结果 句柄 。 为 
了 以 格式 化 的 形式 读 取 输入 数据 ， 我 们 将 那个 结果 句柄 赋 给 用 于 一 个 
DataInputStream 的 构建 器 。DataInputStream 是 我 们 的 最 终 (final) 对 

象 ， 并 是 我 们 进行 读 取 操作 的 接口 。 


在 这 个 例子 中 ， 只 用 到 了 readLine() 方 法 ， 但 理所当然 任何 
DataInputStream 方 法 都 可 以 采用 。 一 旦 抵达 文件 末尾 ，readLine() 就 会 返 
lal—“Snull (F) ， 以 便 中 止 并 退出 while 循 环 。 


“String s2” 用 于 聚集 完整 的 文件 内 容 〈( 包 括 必须 添加 的 新 行 ， 因 为 
readLine() 去 除了 那些 行 )。 随 后 ， 在 本 程序 的 后 面部 分 中 使 用 s2。 最 
后 ， 我 们 调用 close()， 用 它 关 闭 文 件 。 从 技术 上 说 ， 会 在 运行 finalize() 
时 调用 close()。 而 且 我 们 希望 一 旦 程序 退出 ， 就 发 生 这 种 情况 无论 是 
否 进 行 垃圾 收集 ) 。 然 而 ，Java 1.0 有 一 个 非常 突出 的 错误 (Bug) ， 造 
成 这 种 情况 不 会 发 生 。 在 Java 1.1 中 ， 必 须 明确 调用 








System.runFinalizersOnExit(true)， 用 它 保 证 会 为 系统 中 的 每 个 对 象 调用 
finalize0。 然 而 ， 最 安全 的 方法 还 是 为 文件 明确 调用 close()。 


2. 从 内 存 输入 


这 一 部 分 采用 已 经 包含 了 完整 文件 内 容 的 String ”s2， 并 用 它 创建 一 个 
StringBufferInputStream 〈 字 串 绥 冲 输入 流 ) 作为 构建 器 的 参数 ， 要 
求 使 用 一 个 String， 而 非 一 个 StringBuffer) 。 随 后 ， 我 们 用 read0 依 次 读 
取 每 个 字符 ， 并 将 其 发 送 至 控制 台 。 注 意 read0 将 下 一 个 字 节 返回 为 
int， 上 所 以 必须 将 其 造型 为 一 个 char， 以 便 正 确 地 打印 。 


3. 格式 化 内 存 输入 


StringBufferInputStream 的 接口 是 有 限 的 ， 所 以 通常 需要 将 其 封装 到 一 个 
DataInputStream 内 ， 从 而 增强 它 的 能 力 。 然 而 ， 知 选择 用 readByte0 每 次 
读 出 一 个 字符 ， 那 么 所 有 值 都 是 有 效 的 ， 所 以 不 可 再 用 返回 值 来 侦 测 何 
时 结束 输入 。 相 反 ， 可 用 available0) 方 法 判 晰 有 多 少 字符 可 用 。 下 面 这 
个 例子 展示 了 如 何 从 文件 中 一 次 读 出 一 个 字符 : 


//: TestEOF.java 





// Testing for the end of file while reading 
// a byte at a time. 
import java.io.*; 
public class TestEOF { 
public static void main(String[] args) { 
try { 
DataInputStream in = 
new DataInputStream( 
new BufferedInputStream( 


new FileInputStream("TestEof.java"))); 


while(in.available() != 0) 
System.out.print((char)in.readByte()); 
} catch (IOException e) { 


System.err.printin( "IOException" ); 


} 
Lif] l= 


注意 取决 于 当前 从 什么 媒体 读 入 ，avaiable() 的 工作 方式 也 是 有 所 区 别 
的 。 它 在 字面 上 意味 着 “可 以 不 受阻 唉 读 取 的 字 市 数量 *"。 对 一 个 文件 来 
说 ， 它 意味 着 整个 文件 。 但 对 一 个 不 同 种 类 的 数据 流 来 说 ， 它 却 可 能 有 
不 同 的 含义 。 因 此 在 使 用 时 应 考虑 周全 。 


为 了 在 这 样 的 情况 下 侦 测 输入 的 结束 ， 也 可 以 通过 捕获 一 个 违例 来 实 
现 。 然 而 ， 知 真 的 用 违例 来 控制 数据 流 ， 却 显得 有 些 大 材 小 用 。 


4. 行 的 编写 与 文件 输出 


这 个 例子 展示 了 如 何 LineNumberInputStream 来 跟踪 输入 行 的 编号 。 在 这 
里 ， 不 可 简单 地 将 所 有 构建 占 都 组 合 起 来 ， 因 为 必须 保持 
LineNumberInputStream 的 一 个 句柄 (注意 这 并 非 一 种 继承 环境 ， 所 以 不 
能 简单 地 将 in4 造 型 到 一 个 LineNumberInputStream) 。 因 此 ，1i 容 纳 了 指 
问 LineNumberInputStream 的 句柄 ， 然 后 在 它 的 基础 上 创建 一 个 
DataInputStream, 以 便 读 入 数据 。 


这 个 例子 也 展示 了 如 何 将 格式 化 数据 写 入 一 个 文件 。 首 先 创建 了 一 个 
FileOutputStream， 用 它 同 一 个 文件 连接 。 考 虑 到 效率 方面 的 原因 ， 它 生 
成 了 一 个 BufferedOutputStream。 这 几乎 肯定 是 我 们 一 般 的 做 法 ， 但 却 必 
须 明确 地 这 样 做 。 随 后 为 了 进行 格式 化 ， 它 转换 成 一 个 PrintStream。 用 
这 种 方式 创建 的 数据 文件 可 作为 一 个 原始 的 文本 文件 读 取 。 


标志 DataInputStream 何 时 结束 的 一 个 方法 是 readLine()。 一 旦 没有 更 多 的 
字 串 可 以 读 取 ， 它 束 会 返回 null。 每 个 行 都 会 伴随 自己 的 行 写 打 印 到 文 











件 里 。 该 行 号 可 通过 1i 奋 询 。 


可 看 到 用 于 outt 的 、 一 个 明确 指定 的 close0。 知 程序 准备 掉 转 头 来 ， 并 
再 次 读 取 相 同 的 文件 ， 这 种 做 法 就 显得 相当 有 用 。 人 然而， 该 程序 直到 结 
束 也 没有 检查 文件 IODemo.txt。 正 如 以 前 指出 的 那样 ， 如 果 不 为 自己 的 
所 有 输出 文件 调用 close0， 就 可 能 及 现 缓冲 区 不 会 得 到 刷新 ， 造 成 它们 


不 完整 。 。 
10.5.2 输出 流 


两 类 主要 的 输出 流 是 按 它 们 写 入 数据 的 方式 划分 的 : 一 种 按 人 的 习惯 写 
入 ， 另 一 种 为 了 以 后 由 一 个 DataInputStream 而 写 入 。RandomAccessFile 
FEAL, JS FE WBE SUA F DatalnputStream $i 
DataOutputStream. 


5. 保存 与 恢复 数据 


PrintStream 能 格式 化 数据 ， 使 其 能 按 我 们 的 习惯 阅读 。 但 为 了 输出 数 
据 ， 以 便 由 另 一 个 数据 流 恢 复 ， 则 需 用 一 个 DataOutputStream 写 入 数 

据 ， 并 用 一 个 DatainputStream 恢 复 〈 获 取 ) 数据 。 当 然 ， 这 些 数据 流 可 
o 但 这 里 采用 的 是 一 个 文件 ， 并 进行 了 缓冲 处 理 ， 以 加 快 
读 写 速度 。 














注意 字 串 是 用 writeBytes0 写 入 的 ， 而 非 writeCharsO。 知 使 用 后 者 ， 写 入 
的 就 是 16 位 Unicode 字 符 。 由 于 DataInputStream 中 没有 补充 

的 “readChars” 方 法 ， 所 以 不 得 不 用 readChar() 每 次 取出 一 个 字符 。 所 以 
对 ASCII 来 说 ， 更 方便 的 做 法 是 将 字符 作为 字 节 写 入 ， 在 后 面 跟随 一 个 
新 行 ， 然 后 再 用 readLine() 将 字符 当 作 普通 的 ASCII 行 读 回 。 


writeDouble() 将 double 数 字 保 存 到 数据 流 中 ， 并 用 补充 的 readDouble() 恢 
复 它 。 但 为 了 保证 任何 读 方 法 能 够 正常 工作 ， 必 须知 道 数据 项 在 流 中 的 
准确 位 置 ， 因 为 既 有 可 能 将 保存 的 double 数 据 作 为 一 个 简单 的 字 节 序列 
读 入 ， 也 有 可 能 作为 char 或 其 他 格式 读 入 。 所 以 必须 要 么 为 文件 中 的 数 
据 采 用 固定 的 格式 ， 要 么 将 额外 的 信息 保存 到 文件 中 ， 以 便 正确 判断 数 
据 的 存放 位 置 。 


6. 读 写 随机 访问 文件 








正如 早先 指出 的 那样 ，RandomAccessFile 与 IO 层次 结构 的 剩余 部 分 几乎 
是 完全 隔离 的 ， 尺 管 它 也 实现 了 DataInput 和 DataOutput 接 口 。 所 以 不 可 
将 其 与 InputStream 及 OutputStream 子 类 的 任何 部 分 关联 起 来 。 尽 管 也 许 
能 将 一 个 ByteArrayInputStream 当 作 一 个 随机 访问 元 素 对 待 ， 但 只 能 用 
RandomAccessFile 打 开 一 个 文件 。 必 须 假定 RandomAccessFile 已 得 到 了 
正确 的 缓冲 ， 因 为 我 们 不 能 自行 选择 。 


可 以 自行 选择 的 是 第 二 个 构建 器 参数 : 可 决定 以 “只 读 ”(r) 方式 或 “ 读 
写 ”(rw) 方式 打开 一 个 RandomAccessFile 文 件 。 


使 用 RandomAccessFile 的 时 候 ， 类 似 于 组 合 使 用 DataInputStream 和 
DataOutputStream 《因为 它 实现 了 等 同 的 接口 ) 。 除 此 以 外 ， 还 可 看 到 
程序 中 使 用 了 seek0， 以 便 在 文件 中 到 处 移动 ， 对 某 个 值 作出 修改 。 
10.5.3 快捷 文件 处 理 

由 于 以 前 采用 的 一 些 典 型 形式 都 涉及 到 文件 处 理 ， 所 以 大 家 也 许 会 怀疑 
为 什么 要 进行 那么 多 的 代码 输入 一 一 这 正 是 装饰 器 方案 一 个 缺点 。 本 部 
分 将 回 大 家 展示 如 何 创 建 和 使 用 典型 文件 读 取 和 写 入 配置 的 快捷 版 本 。 
这 些 快捷 版 本 均 置 入 packagecom.bruceeckel.tools 中 ( 自 第 5 章 开 始 创 

建 )。 为 了 将 每 个 类 都 添加 到 库 内 ， 只 需 将 其 置 入 适当 的 目录 ， 并 添加 
对 应 的 package 语 句 即 可 。 


7. 快速 文件 输入 


共 想 创建 一 个 对 象 ， 用 它 从 一 个 缓冲 的 DataInputStream 中 读 取 一 个 文 
件 ， 可 将 这 个 过 程 封装 到 一 个 名 为 mnFile 的 类 内 。 如 下 上 所 示 : 


//: InFile.java 





// Shorthand class for opening an input file 
package com.bruceeckel.tools; 

import java.io.*; 

public class InFile extends DataInputStream { 


public InFile(String filename) 


throws FileNotFoundException { 
super ( 
new BufferedInputStream( 
new FileInputStream(filename) )); 
} 
public InFile(File file) 
throws FileNotFoundException { 
this(file.getPath()); 
} 
二 


无 论 构 建 器 的 String 厂 本 还 是 File 版 本 都 包括 在 内 ， 用 于 共同 创建 一 个 


FileInputStream. 


就 象 这 个 例子 展示 的 那样 ， 现 在 可 以 有 效 减 少 创建 文件 时 由 于 重复 强调 
造成 的 问题 。 


8. 快速 输出 格式 化 文件 


亦 可 用 同类 型 的 方法 创建 一 个 PrintStream， 令 其 写 入 一 个 缓冲 文件 。 下 
面 是 对 com.bruceeckel.tools 的 扩展 : 








//: PrintFile.java 


// Shorthand class for opening an output file 
// for human-readable output. 
package com.bruceeckel.tools; 


import java.io.*; 


public class PrintFile extends PrintStream { 
public PrintFile(String filename) 
throws IOException { 
super ( 
new BufferedOutputStream( 
new FileOutputStream( filename) )); 
} 
public PrintFile(File file) 
throws IOException { 
this(file.getPath()); 


} 
} ///:~ 


注意 构建 器 不 可 能 捕获 一 个 由 基础 类 构建 器 “ 掷 ?出 的 违例 。 
9. 快速 输出 数据 文件 


最 后 ， 利 用 类 似 的 快捷 方式 可 创建 一 个 缓冲 输出 文件 ， 用 它 保 存 数 据 
(与 由 人 观看 的 数据 格式 相反 》: 


//: OutFile.java 


// Shorthand class for opening an output file 
// for data storage. 

package com.bruceeckel.tools; 

import java.io.*; 


public class OutFile extends DataOutputStream { 


public OutFile(String filename) 
throws IOException { 
super ( 

new BufferedOutputStream( 
new FileOutputStream( filename) )); 

} 

public OutFile(File file) 
throws IOException { 
this(file.getPath()); 

} 

} ///:~ 


非常 奇怪 的 是 〈 也 非常 不 幸 ) ，Java 库 的 设计 者 居然 没 想 到 将 这 些 便利 
音 施 直接 作为 他 们 的 一 部 分 标准 提供 。 


10.5.4 从 标准 输入 中 读 取 数据 


以 Unix 首 先 倡导 的 “标准 输入 入 “标准 输出 ?以 及 “标准 错误 输出 ”概念 为 
基础 ，Java 提 供 了 相应 的 System.in，System.out 以 及 System.err。 贯 这 一 
整 本 书 ， 大 家 都 会 接触 到 如 何 用 System.out 进 行 标准 输出 ， 它 已 预 封装 
成 一 个 PrintStream 对 象 。System.err 同 样 是 一 个 PrintStream， 但 System.in 
是 一 个 原始 的 InputStream， 末 进行 任何 封装 处 理 。 这 意味 着 尽管 能 直接 
使 用 System.out 和 System.err， 但 必须 事先 封装 System.in， 人 否则 不 能 从 中 
读 取 数据 。 


典型 情况 下 ， 我 们 希望 用 readLine() 每 次 读 取 一 行 输 入 信息 ， 所 以 需要 
将 System.in 封 装 到 一 个 DataInputStream 中 。 这 是 Java 1.0 进 行 行 输入 时 采 
Ae INE. FER, AZUSA Java 1.1 的 解决 方案 。 下 面 是 
个 简单 的 例子 ， 作 用 是 回应 我 们 键入 的 每 一 行内 容 : 


//: Echo.java 











// How to read from standard input 
import java.io.*; 
public class Echo { 
public static void main(String[] args) { 
DataInputStream in = 
new DataInputStream( 
new BufferedInputStream(System.in)); 
String s; 
try { 
while((s = in.readLine()).length() != 0) 
System.out.println(s); 
// An empty line terminates the program 
} catch(IOException e) { 


e.printStackTrace(); 


} 


} ///:~ 
之 所 以 要 使 用 try 块 ， 是 由 于 readLineO 可 能 “ 掷 ” 出 一 个 IOException。 注 
意 同 其 他 大 多 数 流 一 样 ， 也 应 对 System.in 进 行 缓冲 。 


由 于 在 每 个 程序 中 都 要 将 System.in 封 装 到 一 个 DataInputStream 内 ， 所 以 
显得 有 点 不 方便 。 但 采用 这 种 设计 方案 ， 可 以 获得 最 大 的 灵活 性 。 








10.5.5 管道 数据 流 


本 章 已 简要 介绍 了 PipedInputStream 〈 管 道 输 入 流 ) 和 
PipedOutputStream 〈 管 道 输出 流 ) 。 尽 管 描 述 不 十 分 详细 ， 但 并 不 是 说 
它们 作用 不 大 。 然 而 ， 只 有 在 掌握 了 多 线程 处 理 的 概念 后 ， 才 可 真正 体 
会 它们 的 价值 所 在 。 原 因 很 简单 ， 因 为 管道 化 的 数据 流 就 是 用 于 线程 之 
间 的 通信 。 这 方面 的 问题 将 在 第 14 章 用 一 个 示例 说 明 。 








10.6 StreamTokenizer 


尽管 StreamTokenizer 并 不 是 从 InputStream 或 OutputStream 衍 生 的 ， 但 它 
只 随同 mputStream 工 作 ， 所 以 十 分 恰当 地 包括 在 库 的 IO 部 分 中 。 


StreamTokenizer 类 用 于 将 任何 InputStream 分 割 为 一 系列 “ 记 

+5” (Token) 。 这 些 记 号 实际 是 一 些 断 续 的 文本 块 ， 中 间 用 我 们 选择 的 
任何 东西 分 隔 。 例 如 ， 我 们 的 记号 可 以 是 单词 ， 中 间 用 空白 《空格 ) 以 
及 标点 符号 分 隔 。 


用 于 计算 各 个 单词 在 文本 文件 中 重复 出 现 的 次 











//: SortedWordCount. java 


// Counts words ina file, outputs 
// results in sorted form. 
import java.io.*; 
import java.util.*; 
import c08.*; // Contains StrSortVector 
class Counter { 
private int i = 1; 
int read() { return i; } 
void increment() { i++; } 
} 
public class SortedwordCount { 
private FileInputStream file; 


private StreamTokenizer st; 


private Hashtable counts = new Hashtable(); 
SortedwWordCount(String filename) 
throws FileNotFoundException { 
try { 
file = new FileInputStream(filename) ; 
st = new StreamTokenizer(file); 
st.ordinaryChar('.'); 
st.ordinaryChar('-'); 
} catch(FileNotFoundException e) { 
System.out.printin( 
"Could not open " + filename); 


throw e; 


} 


void cleanup() { 
try { 
file.close(); 
} catch(IOException e) { 
System.out.printin( 


"file.close() unsuccessful"); 


} 


void countwords() { 


try { 


while(st.nextToken() != 
StreamTokenizer.TT_EOF) { 
String s; 
Switch(st.ttype) { 
case StreamTokenizer.TT_EOL: 
s = new String("EOL"); 
break; 
case StreamTokenizer.TT_NUMBER: 
s = Double.toString(st.nval); 
break; 
case StreamTokenizer.TT_WORD: 
s = st.sval; // Already a String 
break; 
default: // single character in ttype 
s = String.valueOf((char)st.ttype); 
} 
if (counts.containsKey(s) ) 
((Counter )counts.get(s)).increment(); 
else 
counts.put(s, new Counter()); 


} 


} catch(IOException e) { 


System.out .println( 


"st.nextToken() unsuccessful"); 


} 


Enumeration values() { 
return counts.elements(); 
J 
Enumeration keys() { return counts.keys(); } 
Counter getCounter(String s) { 
return (Counter )counts.get(s); 
} 
Enumeration sortedKeys() { 
Enumeration e = counts.keys(); 
StrSortVector sv = new StrSortvVector(); 
while(e.hasMoreElements() ) 
sv.addElement((String)e.nextElement()); 
// This call forces a sort: 
return sv.elements(); 
} 
public static void main(String[] args) { 
try { 
SortedwWordCount wc = 


new SortedWordCount(args[0]); 


wc .countWwords(); 
Enumeration keys = wc.sortedKeys(); 
while(keys.hasMoreElements()) { 
String key = (String)keys.nextElement(); 
System.out.println(key + ": " 
+ wc.getCounter(key).read()); 
} 
wc .cleanup( ); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
TTT 


最 好 将 结果 按 排 序 格式 输出 ， 但 由 于 Java 1.0 和 Java 1.1 都 没有 提供 任何 
排序 方法 ， 所 以 必须 由 目 己 动手 。 这 个 目标 可 用 一 个 StrSortVector 方 便 

地 达成 〈 创 建 于 第 8 章 ， 属 于 那 一 章 创建 的 软件 包 的 一 部 分 。 记 住 本 书 

所 有 了 目录 的 起 始 目录 都 必须 位 于 类 路 径 中 ， 人 否则 程序 将 不 能 正确 地 编 
译 ) 。 


为 打开 文件 ， 使 用 了 一 个 FileInputStream。 而 且 为 了 将 文件 转换 成 单 
词 ， 从 FileInputStream 中 创建 了 一 个 StreamTokenizer。 在 
StreamTokenizer 中 ， 存 在 一 个 默认 的 分 隅 符 列 表 ， 我 们 可 用 一 系列 方法 
加 入 更 多 的 分 隔 符 。 在 这 里 ， 我 们 用 ordinaryChar0 指 出 “该 字符 没有 特 
别 重 要 的 意义 ?”， 所 以 解析 器 不 会 把 它 当 作 上 自己 创建 的 任何 单词 的 一 部 
分 。 例 如 ，st.ordinaryChar(.) 表 示 小 数 点 不 会 成 为 解析 出 来 的 单词 的 一 
部 分 。 在 与 Java 配 套 提供 的 联机 文档 中 ， 可 以 找到 更 多 的 相关 信息 。 


在 countWords() 中 ， 每 次 从 数据 流 中 取出 一 个 记号 ， 而 ttype 信 息 的 作用 





是 判断 对 每 个 记号 采取 什么 操作 一 一 因为 记号 可 能 代表 一 个 行 尾 、 一 个 
数字 、 一 个 字 串 或 者 一 个 字符 。 


找到 一 个 记号 后 ， 会 查询 Hashtable counts， 核 实 其 中 是 否 已 经 
DE” (Key) 的 形式 包含 了 一 个 记号 。 知 答案 是 肯定 的 ， 对 应 的 
Counter GFA) 对象 束 会 增值 ， 指 出 已 找到 该 单词 的 另 一 个 实例 。 

咎 答 案 为 否 ， 则 新 建 一 个 Counter 为 Counter 构 建 器 会 将 它 的 值 初 
始 化 为 1， 正 是 我 们 计算 单词 数量 时 的 要 求 。 


SortedWordCount 并 不 属于 Hashtable《〈 散 列表 ) 的 一 种 类 型 ， 所 以 它 不 
会 继承 。 它 执行 的 一 种 特定 类 型 的 操作 ， 所 以 尽管 keys0 和 values() 方 法 
都 必须 重新 揭示 出 来 ， 但 仍 不 表示 应 使 用 那个 继承 ， 因 为 大 量 Hashtable 
方法 在 这 里 都 是 不 适当 的 。 除 此 以 外 ， 对 于 另 一 些 方法 来 说 《比如 
getCounter( 一 一 用 于 获得 一 个 特定 字 串 的 计数 器 ， 又 如 sortedKeys() 
本 于 产生 一 个 枚 举 ) ， 它 们 最 终 都 改变 了 SortedWordCount 接 口 的 
形 却 。 


在 main() 内 ， 我 们 用 SortedWordCount 打 开 和 计算 文件 中 的 单词 数量 
总 共 只 用 了 两 行 代码 。 随 后 ， 我 们 为 一 个 排 好 序 的 键 (单词 ) 列表 提取 
出 一 个 枚 举 。 并 用 它 获 得 每 个 键 以 及 相关 的 Count (计数 ) 。 注 意 必 须 
调用 deanup()， 奋 则 文件 不 能 正常 关闭 。 


采用 了 StreamTokenizer 的 第 二 个 例子 将 在 第 17 章 提供 。 





























10.6.1 StringTokenizer 


尽管 并 不 必要 IO 库 的 一 部 分 ， 但 StringTokenizer 提 供 了 与 
StreamTokenizer 极 相似 的 功能 ， 所 以 在 这 里 一 并 讲述 。 


StringTokenizer 的 作用 是 每 次 返回 字 串 内 的 一 个 记号 。 这 些 记 号 是 一 些 
由 制 表 站 、 空 格 以 及 新 行 分 隔 的 连续 字符 。 因 此 ， 字 串 “wWhere is my 
cat?22” 的 记号 分 别 是 “Where” “is”, “my” 和 “cat?”。 与 StreamTokenizer 类 
似 ， 我 们 可 以 指示 StringTokenizer 按 照 我 们 的 愿望 分 割 输入 。 但 对 于 
StringTokenizer， 却 需要 同 构 建 器 传递 男 一 个 参数 ， 即 我 们 想 使 用 的 分 
隔 字 串 。 通 种 ， 如 果 想 进行 更 复杂 的 操作 ， 应 使 用 StreamTokenizer。 


可 用 nextToken0 回 StringTokenizer 对 象 请 求 字 串 内 的 下 一 个 记 扎 。 该 方 
法 要 么 返回 一 个 记号 ， 要 么 返回 一 个 空 字 串 〈 表 示 没 有 记号 剩 下 ) 。 





作为 一 个 例子 ， 下 述 程序 将 执行 一 个 有 限 的 句法 分 机 ， 碍 询 键 短语 序 
列 ， 了 解 句子 暗示 的 是 快乐 亦 或 起 伤 的 含义 。 


//: AnalyzeSentence , java 





// Look for particular sequences 
// within sentences. 
import java.util.*; 
public class AnalyzeSentence { 
public static void main(String[] args) { 
analyze("I am happy about this"); 
analyze("I am not happy about this"); 
analyze("I am not! I am happy"); 
analyze("I am sad about this"); 
analyze("I am not sad about this"); 
analyze("I am not! I am sad"); 
analyze("Are you happy about this?"); 
analyze("Are you sad about this?"); 
analyze("It's you! I am happy"); 
analyze("It's you! I am sad"); 
} 
static StringTokenizer st; 
static void analyze(String s) { 


prt("\nnew sentence >> " + s); 


boolean sad = false; 
st = new StringTokenizer(s); 
while (st.hasMoreTokens()) { 
String token = next(); 
// Look until you find one of the 
// two starting tokens: 
if(!token.equals("I") && 
!'token.equals("Are") ) 
continue; // Top of while loop 
if(token.equals("I")) { 
String tk2 = next(); 
if(!tk2.equals("am")) // Must be after I 
break; // Out of while loop 
else { 
String tk3 = next(); 
if(tk3.equals("sad")) { 
sad = true; 
break; // Out of while loop 
} 
if (tk3.equals("not")) { 
String tk4 = next(); 
if(tk4.equals("sad") ) 


break; // Leave sad false 


if(tk4.equals("happy")) { 
sad = true; 


break; 


} 


if(token.equals("Are")) { 
String tk2 = next(); 
if(!tk2.equals("you") ) 
break; // Must be after Are 
String tk3 = next(); 
if(tk3.equals("sad") ) 
sad = true; 


break; // Out of while loop 


} 
if(sad) prt("Sad detected"); 
Í 
static String next() { 
if(st.hasMoreTokens()) { 


String s = st.nextToken(); 


prt(s); 


return s; 


} 
else 


return ""; 


} 
static void prt(String s) { 
System.out.println(s); 


} 
SS 


对 于 准备 分 析 的 每 个 字 串 ， 我 们 进入 一 个 while 循 环 ， 并 将 记号 从 那个 
字 串 中 取出 。 请 注意 第 一 个 诗 语句 ， 假 如 记号 既 不 是 “二 ， 也 不 

是 “Are”， 就 会 执行 continue 〈 返 回 循环 起 点 ， 再 一 次 开始 ) 。 这 意味 着 
除非 发 现 一 个 “或 者 “Are”， 才 会 真正 得 到 记号 。 大 家 可 能 想 用 == 代 蔡 
equals(0) 方 法 ， 但 那样 做 会 出 现 不 正常 的 表现 ， 因 为 == 比 较 的 是 句柄 
值 ， 而 equals0 比 较 的 是 内 容 。 


analyze() 方 法 剩余 部 分 的 逻辑 是 搜索 “I_ am ”sad”( 我 很 忧伤 、“I am 
nothappy”( 我 不 快乐 ) 或 者 “Are you sad?”( 你 悲伤 蚂 ? ) 这 样 的 句法 
格式 。 知 没有 break 语 句 ， 这 方面 的 代码 甚至 可 能 更 加 散乱 。 大 家 应 注 
意 对 一 个 典型 的 解析 器 来 说 ， 通 常 都 有 这 些 记 号 的 一 个 表格 ， 并 能 在 读 
取 新 记 写 的 时 候 用 一 小 段 代 码 在 表格 内 移动 。 


无 论 如 何 ， 只 应 将 StringTokenizer 看 作 StreamTokenizer 一 种 简单 而 且 特 
殊 的 简化 形式 。 然 而 ， 如 果 有 一 个 字 串 需要 进行 记号 处 理 ， 而 且 
StringTokenizer 的 功能 实在 有 限 ， 那 么 应 该 做 的 全 部 事情 就 是 用 
StringBufferInputStream 将 其 转换 到 一 个 数据 流 里 ， 再 用 它 创建 一 个 功能 
更 强大 的 StreamTokenizer。 

















10.7 Java 1.1 的 IO 流 


到 这 个 时 候 ， 大 家 或 许 会 陷入 一 种 困境 之 中 ， 怀 疑 是 否 存在 IO 流 的 另 一 
种 设计 方案 ， 并 可 能 要 求 更 大 的 代码 量 。 还 有 人 能 提出 一 种 更 古怪 的 设 
计 吗 ?事实 上 ，Java 1.1 对 IO 流 库 进行 了 一 些 重大 的 改进 。 看 到 Reader 和 
Writer 类 时 ， 大 多 数 人 的 第 一 个 印象 (就 象 我 一 样 ) 就 是 它们 用 来 将 换 
原来 的 mputStream 和 OutputStream 类 。 但 实情 并 非 如 此 。 尽 管 不 建议 使 
用 原始 数据 流 库 的 某 些 功 能 (如 使 用 它们 ， 会 从 编译 器 收 到 一 条 警告 消 
轧 ) ， 但 原来 的 数据 流 依 然 得 到 了 保留 ， 以 便 维 持 向 后 兼容 ， 而 且 : 


(1) 在 老式 层次 结构 里 加 入 了 新 类 ， 所 以 Sun 公 司 明显 不 会 放 莽 老式 数据 
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(2) 在 许多 情况 下 ， 我 们 需要 与 新 结构 中 的 类 联合 使 用 老 结构 中 的 类 。 
为 达到 这 个 目的 ， 需 要 使 用 一 些 “ 桥 ”类 : InputStreamReader 将 一 个 
InputStream 转 换 成 Reader，OutputStreamWriter 将 一 个 OutputStream 转 换 
成 Writer。 


所 以 与 原来 的 IO 流 库 相 比 ， 经 都 要 对 新 IO 流 进行 层次 更 多 的 封装 。 同 
样 地 ， 这 也 属于 装饰 堪 方 案 的 一 个 缺点 一 一 需要 为 额外 的 灵活 性 付出 代 


价 。 


之 所 以 在 Java 1.1 里 添加 了 Reader 和 Writer 层 次 ， 最 重要 的 原因 便 是 国际 
化 的 需求 。 老 式 IO 流 层次 结构 只 文 持 8 位 字 节 流 ， 不 能 很 好 地 控制 16 位 
Unicode 字 符 。 由 于 Unicode 主 要 面 癌 的 是 国际 化 文 持 〈Java 内 含 的 char 
是 16 位 的 Unicode) ， 所 以 添加 了 Reader 和 Writer 层 次 ， 以 提供 对 所 有 IO 
操作 中 的 Unicode 的 支持 。 除 此 之 外 ， 新 库 也 对 速度 进行 了 优化 ， 可 比 
旧 库 更 快 地 运行 。 


与 本 书 其 他 地 方 一 样 ， 我 会 试 着 提供 对 类 的 一 个 概述 ， 但 假定 你 会 利用 
联机 文档 搞定 所 有 的 细节 ， 比 如 方法 的 详尽 列表 等 。 
10.7.1 数据 的 发 起 与 接收 


Java ”1.0 的 几乎 所 有 IO 流 类 都 有 对 应 的 Java ”1.1 类 ， 用 于 提供 内 建 的 
Unicode 管 理 。 似 乎 最 容易 的 事情 就 是 “全 部 使 用 新 类 ， 再 也 不 要 用 旧 
的 ”， 但 实际 情况 并 没有 这 么 简单 。 有 些 时 候 ， 由 于 受到 库 设 计 的 一 些 





























限制 ， 我 们 不 得 不 使 用 Java 1.0 的 IO 流 类 。 特 别 要 指出 的 是 ， 在 旧 流 库 
的 基础 上 新 加 了 java.util.zip 库 ， 它 们 依赖 旧 的 流 组 件 。 所 以 最 明智 的 做 
法 是 “尝试 性 ”地 使 用 Reader 和 Writer 类 。 若 代码 不 能 通过 编译 ， 便 知道 
必须 换 回 老式 库 。 


下 面 这 张 表格 分 旧 库 与 新 库 分 别 总 结 了 信息 发 起 与 接收 之 间 的 对 应 关 
Sources & Sinks: Corresponding Java 1.1 class 
Java 1.0 class 

Reader 

converter: InputStreamReader 


Writer 


converter: OutputStreamWriter 


FileReader 


StringReader 
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10.7.2 修改 数据 流 的 行为 


在 Java 1.0 中 ， 数 据 流 通过 FiltermputStream 和 FilterOutputStream 的 “装饰 
器 ”(Decorator) 子 类 适应 特定 的 需求 。Java ”1.1 的 IO 流 沿用 了 这 一 思 
想 ， 但 没有 继续 采用 所 有 装饰 融 都 从 相同 “filter”( 过 滤器， 基础 类 中 衍 

EE E Ce erie sy ev 
的 困惑 。 


在 下 面 这 张 表 格 中 ， 对 应 关系 比 上 一 张 表 要 粗粮 一 些 。 之 所 以 会 出 现 这 
个 差别 ， 是 由 类 的 组 织造 成 的 : 尽管 BufferedOutputStream 是 
FilterOutputStream 的 一 个 子 类 ， 但 是 BufferedWriter 并 不 是 FilterWriter 的 
子 类 《对 后 者 来 说 ， 尽 管 它 是 一 个 抽象 类 ， 但 没有 自己 的 子 类 或 者 近似 
子 类 的 东西 ， 也 没有 一 个 “ 占 位 符 ” 可 用 ， 所 以 不 必 费 心地 寻找 ) 。 然 
而 ， 两 个 类 的 接口 是 非常 相似 的 ， 而 且 不 管 在 什么 情况 下 ， 显 然 应 该 尽 
可 能 地 使 用 新 版 本 ， 而 不 应 考虑 旧版 本 (也 就 是 说 ， 除 非 在 一 些 类 中 必 
须 生 成 一 个 Stream， 不 可 生成 Reader 或 者 Writer) 。 























Filters: 


Corresponding Java 1.1 class 
Java 1.0 class 


FilterInputStream FilterReader 


FilterOutputStream FilterWriter (abstract class with no 
subclasses) 





BufferedInputStream BufferedReader 





(also has readLine( )) 


BufferedOutputStream |BufferedWriter 
en WC sa 


DataInputStream use DataInputStream 


(Except when you need to use readLine( ), 
when you should use a BufferedReader) 





LineNumberInputStream}/LineNumberReader 


StreamTokenizer StreamTokenizer 


(use constructor that takes a Reader instead) 


PushBackInputStream liPushBackReader 








过 滤器 : Java 1.0% 对 应 的 Java 1.128 


FilterInputStream FilterReader 

FilterOutputStream FilterWriter (没有 子 类 的 抽象 类 ) 
BufferedInputStream BufferedReader (也 有 readLine()) 
BufferedOutputStream BufferedWriter 


DataInputStream 使 用 DataInputStream (除非 要 使 用 readLine()， 那 时 需要 
使 用 一 个 BufferedReader) 


PrintStream PrintWriter 

LineNumberInputStream LineNumberReader 
StreamTokenizer StreamTokenizer (用 构建 右 取 代 Reader) 
PushBackInputStream PushBackReader 


有 一 条 规律 是 显然 的 : 在 想 使 用 readLineO0， 束 不 要 再 用 一 个 


DataInputStream 来 实现 〈 人 否则 会 在 编译 期 得 到 一 条 出 错 消 轧 ) ， 而 应 使 
用 一 个 BufferedReader。 但 除 这 种 情况 以 外 ，DataInputStream 仍 是 Java 
1.1 IO 库 的 “首选 成员。 


为 了 将 加 PrintWriter 的 过 渡 变 得 更 加 自然 ， 它 提供 了 能 采用 任何 
OutputStream 对 象 的 构建 器 。PrintWriter 提 供 的 格式 化 支持 没有 
PrintStream 那 么 多 ; 但 接口 几乎 是 相同 的 。 

10.7.3 未 改变 的 类 


显然 ，Java 库 的 设计 人 员 和 觉得 以 前 的 一 些 类 坚 无 问题 ， 所 以 没有 对 它们 
作 任何 修改 ， 可 象 以 前 那样 继续 使 用 它们 : 


没有 对 应 Java 1.1 类 的 Java 1.0 类 





DataOutputStream 
File 
RandomAccessFile 


SequencelInputStream 


特别 未 加 改动 的 是 DataOutputStream， 所 以 为 了 用 一 种 可 转移 的 格式 保 
存 和 获取 数据 ， 必 须 沿 用 InputStream 和 OutputStream 层 次 结构 。 


10.7.4 一 个 例子 


为 体验 新 类 的 效果 ， 下 面 让 我 们 看 看 如 何 修改 IOStreamDemo.java 示 例 
的 相应 区 域 ， 以 便 使 用 Reader 和 Writer 类 : 


//: NewIODemo. java 


// Java 1.1 IO typical usage 
import java.io.*; 


public class NewIODemo { 


public static void main(String[] args) { 
try { 

// 1. Reading input by lines: 
BufferedReader in = 

new BufferedReader ( 

new FileReader(args[0])); 

String s, s2 = new String(); 
while((s = in.readLine())!= null) 

s2 += s + "\n"; 
in.close(); 
// 1b. Reading standard input: 
BufferedReader stdin = 

new BufferedReader ( 

new InputStreamReader(System.in)); 

System.out.print("Enter a line:"); 
System.out.printin(stdin.readLine()); 
// 2. Input from memory 
StringReader in2 = new StringReader(s2); 
int c; 
while((c = in2.read()) != -1) 

System.out.print((char)c); 


// 3. Formatted memory input 


try { 


DataInputStream in3 = 

new DataInputStream( 

// Oops: must use deprecated class: 
new StringBufferInputStream(s2) ); 

while(true) 

System.out.print((char)in3.readByte()); 

} catch(EOFException e) { 

System.out.println("End of stream"); 


} 


// 4. Line numbering & file output 
try { 
LineNumberReader 1i = 
new LineNumberReader ( 
new StringReader(s2)); 
BufferedReader in4 = 
new BufferedReader (li); 
PrintWriter out1 = 
new Printwriter ( 
new Bufferedwriter ( 
new FileWriter("IODemo.out"))); 
while((s = in4.readLine()) != null ) 
outi1.printin( 


"Line " + 1i.getLineNumber() + s); 


try { 


} 


} 


outi.close(); 
catch(EOFException e) { 


System.out.printin("End of stream"); 


// 5. Storing & recovering data 


DataOutputStream out2 = 
new DataOutputStream( 
new BufferedOutputStream( 
new FileOutputStream("Data.txt"))); 
out2.writeDouble(3.14159); 
out2.writeBytes("That was pi"); 
out2.close(); 
DataInputStream in5 = 
new DataInputStream( 
new BufferedInputStream( 
new FileInputStream("Data.txt"))); 
BufferedReader in5br = 
new BufferedReader ( 
new InputStreamReader(in5) ); 
// Must use DataInputStream for data: 
System.out.printin(in5.readDouble()); 


// Can now use the "proper" readLine(): 


System.out.printin(in5br.readLine()); 
} catch(EOFException e) { 
System.out.printin("End of stream"); 
} 
// 6. Reading and writing random access 
// files is the same as before. 
// (not repeated here) 
} catch(FileNotFoundException e) { 
System.out.println( 
"File Not Found:" + args[1]); 
} catch(IOException e) { 


System.out.println("IO Exception"); 


} 
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大 家 一 般 看 见 的 是 转换 过 程 非常 直观 ， 代 码 看 起 来 也 颇 相 似 。 但 这 些 都 
C i 最 重要 的 是 ， 由 于 随机 访问 文件 已 经 改变 ， 所 以 第 6 
W à 


第 1 市 收缩 了 一 点 儿 ， 因 为 假如 要 做 的 全 部 事情 就 是 读 取 行 输入 ， 那 么 
只 需要 将 一 个 FileReader 封 装 到 BufferedReader 之 内 即 可 。 第 1b 节 展示 了 
封装 System.in， 以 便 恋 取 控 制 台 输入 的 新 方法 。 这 里 的 代码 量 增多 了 一 
些 ， 因 为 System.in 是 一 个 DataInputStream， 而 且 BufferedReader 需 要 一 个 
Reader 参 数 ， 所 以 要 用 InputStreamReader 来 进行 转换 。 


在 2 节 ， 可 以 看 到 如 果 有 一 个 字 串 ， 而 且 想 从 中 读 取 数据 ， 只 需 用 一 个 
StringReader 蔡 换 StringBufferInputStream， 剩 下 的 代码 是 完全 相同 的 。 








第 3 节 揭 示 了 新 IO 流 库 设计 中 的 一 个 错误 。 如 果 有 一 个 字 串 ， 而 且 想 从 
中 该 取 数 据 ， 那 么 不 能 再 以 任何 形式 使 用 StringBufferInputStream。 知 编 
译 一 个 涉及 StringBufferInputStream 的 代码 ， 会 得 到 一 条 “反对 ?消息 ， 告 
诉 我 们 不 要 用 它 。 此 时 最 好 换 用 一 个 StringReader。 但 是 ， 假 如 要 象 第 3 
节 这 样 进行 格式 化 的 内 存 输入 ， 束 必须 使 用 DataImnputStream 一 没有 什 
么 “DataReader” 可 以 代替 它 而 DataInputStream 很 不 幸 地 要 求 用 到 一 
个 InputStream 参 数 。 所 以 我 们 没有 选择 的 余地 ， 只 好 使 用 编译 器 不 赞成 
的 StringBufferInputStream 类 。 编 译 器 同样 会 发 出 反对 信息 ， 但 我 们 对 此 
束手无策 (注释 @)。 


StringReader 蔡 换 StringBufferInputStream， 剩 下 的 代码 是 完全 相同 的 。 
D: 到 你 现在 正式 使 用 的 时 候 ， 这 个 错误 可 能 已 经 修正 。 


第 4 节 明 显 是 从 老式 数据 流 到 新 数据 法 的 一 个 直接 转换 ， 没 有 需要 特别 
指出 的 。 在 第 5 市 中 ， 我 们 被 强迫 使 用 所 有 的 老式 数据 流 ， 因 为 
DataOutputStream 和 DataInputStream 要 求 用 到 它们 ， 而 且 没 有 可 供 蔡 换 
的 东西 。 然 而 ， 编 译 期 间 不 会 产生 任何 “反对 ”信息 。 大 不 赞成 一 种 数据 
流 ， 通 常 是 由 于 它 的 构建 器 产生 了 一 条 反对 消息 ， 禁 止 我 们 使 用 整个 
类 。 但 在 DataInputStream 的 情况 下 ， 只 有 readLine0) 是 不 赞成 使 用 的 ， 
为 我 们 最 好 为 readLine() 使 用 一 个 BufferedReader (但 为 其 他 所 有 格式 化 
输入 都 使 用 一 个 DataInputStream) 。 


若 比 较 第 5 节 和 IOStreamDemo.java 中 的 那 一 小 节 ， 会 注意 到 在 这 个 版 本 
中 ， 数 据 是 在 文本 之 前 写 入 的 。 那 是 由 于 Java 1.1 本 身 存在 一 个 错误 ， 
如 下 述 代 人 码 所 示 : 


//: TOBug.java 
































// Java 1.1 (and higher?) IO Bug 
import java.io.*; 
public class I0Bug { 
public static void main(String[] args) 


throws Exception { 


DataOutputStream out = 
new DataOutputStream( 
new BufferedOutputStream( 
new FileOutputStream("Data.txt"))); 
out .writeDouble(3.14159); 
out.writeBytes("That was the value of pi\n"); 
out.writeBytes("This is pi/2:\n"); 
out .writeDouble(3.14159/2) ; 
out.close(); 
DataInputStream in = 
new DataInputStream( 
new BufferedInputStream( 
new FileInputStream("Data.txt"))); 
BufferedReader inbr = 
new BufferedReader ( 
new InputStreamReader(in)); 
// The doubles written BEFORE the line of text 
// read back correctly: 
System.out.println(in.readDouble()); 
// Read the lines of text: 
System.out.println(inbr.readLine()); 
System.out.println(inbr.readLine()); 


// Trying to read the doubles after the line 


// produces an end-of-file exception: 
System.out.println(in.readDouble()); 


} 
} ///:~ 





看 起 来 ， 我 们 在 对 一 个 writeBytes0) 的 调用 之 后 写 入 的 任何 东西 部 不 是 能 
够 恢复 的 。 这 是 一 个 十 分 有 限 的 错误 ， 硕 望 在 你 读 到 本 书 的 时 候 已 获得 
改正 。 为 检测 是 否 改 正 ， 请 运行 上 述 程序 。 奉 没有 得 到 一 个 违例 ， 而 且 
值 都 能 正确 打印 出 来 ， 束 表明 已 经 改正 。 


10.7.5 重 导 向 标准 IO 


Java 1.1 在 System 类 中 添加 了 特殊 的 方法 ， 人 允许 我 们 重新 定向 标准 输 
入 、 输 出 以 及 错误 IO 流 。 此 时 要 用 到 下 述 简单 的 静态 方法 调用 : 











setIn(InputStream) 
setOut(PrintStream) 


setErr(PrintStream) 

如 果 突 然 要 在 屏幕 上 生成 大 量 输出 ， 而 且 滚动 的 速度 快 于 人 们 的 阅读 速 
度 ， 输 出 的 重 定向 就 显得 特别 有 用 。 在 一 个 命令 行程 序 中 ， 如 果 想 重复 
测试 一 个 特定 的 用 户 输 入 序列 ， 输 入 的 重 定 同 也 显得 特别 有 价值 。 下 面 
这 个 简单 的 例子 展示 了 这 些 方法 的 使 用 : 


//: Redirecting.java 








// Demonstrates the use of redirection for 
// standard IO in Java 1.1 
import java.io.*; 


class Redirecting { 


public static void main(String[] args) { 
try { 
BufferedInputStream in = 
new BufferedInputStream( 
new FileInputStream( 
"Redirecting.java")); 
// Produces deprecation message: 
PrintStream out = 
new PrintStream( 


new BufferedOutputStream( 


new FileOutputStream("test.out"))); 


System.setiIn(in); 
System.setOut(out); 
System.setErr(out); 
BufferedReader br = 

new BufferedReader ( 

new InputStreamReader(System.in)); 

String s; 
while((s = br.readLine()) != null) 

System.out.println(s); 
out.close(); // Remember this! 

} catch(IOException e) { 


e.printStackTrace(); 


} 
FA ss 


这 个 程序 的 作用 是 将 标准 输入 同一 个 文件 连接 起 来 ， 并 将 标准 输出 和 错 
REE RAT ME 
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Note:The constructor java.io.PrintStream(java.io.OutputStream) has been 
deprecated. 


注意 : 不 推荐 使 用 构建 器 java.io.PrintStream (java.io.OutputStream) 。 


然而 ， 无 论 System.setOut0 还 是 System.setErrO 都 要 求 用 一 个 PrintStream 
作为 参数 使 用 ， 所 以 必须 调用 PrintStream 构 建 器 。 所 以 大 家 可 能 会 觉得 
奇怪 ， 既 然 Java 1.1 通 过 反对 构建 器 而 反对 了 整个 PrintStrream， 为 什么 库 
的 设计 人 员 在 琴 加 这 个 反对 的 同时 ， 依 然 为 System 添加 了 新 方法 ， 且 指 
明 要 求 用 PrintStream， 而 不 是 用 PrintWriter 呢 ? 毕竟 ， 后 者 是 一 个 轿 新 
和 首选 的 蔡 换 措施 呀 ? 这 真 令 人 费解 。 





10.8 压缩 


Java ”1.1 也 添加 一 个 类 ， 用 以 支持 对 压缩 格式 的 数据 流 的 读 写 。 它 们 封 
装 到 现成 的 IO 类 中 ， 以 提供 压缩 功能 。 


此 时 Java 1.1 的 一 个 问题 显得 非常 突出 : 它们 不 是 从 新 的 Reader 和 Writer 
类 衍生 出 来 的 ， 而 是 属于 InputStream 和 OutputStream 层 次 结构 的 一 部 

分 。 所 以 有 时 不 得 不 混合 使 用 两 种 类 型 的 数据 流 《〈 注 意 可 用 
InputStreamReader 和 OutputStreamWriter 在 不 同 的 类 型 间 方 便 地 进行 转 

换 ) 。 








Java 1.1 压 缩 类 功能 


CheckedInputStream GetCheckSumO 为 任何 InputStream 产 生 校 验 和 《不仅 
是 解压 ) 


CheckedOutputStream GetCheckSum() 为 任何 OutputStream 产 生 校 验 和 
《不 仅 是 解压 ) 


DeflaterOutputStream 用 于 压缩 类 的 基础 类 
ZipOutputStream 一 个 DeflaterOutputStream， 将 数据 压缩 成 Zip 文 件 格 式 


GZIPOutputStream 一 个 DeflaterOutputStream， 将 数据 压缩 成 GZIP 文 件 格 
式 


InflaterInputStream 用 于 解压 类 的 基础 类 
ZipInputStream 一 个 DeflaterInputStream， 解 压 用 Zip 文 件 格 式 保存 的 数据 


GZIPInputStream 一 个 DeflaterInputStream， 解 压 用 GZIP 文 件 格 式 保 存 的 
数据 


尽管 存在 许多 种 压缩 算法 ， 但 是 Zip 和 GZIP 可 能 最 常用 的 。 所 以 能 够 很 
方便 地 用 多 种 现成 的 工具 来 读 写 这 些 格式 的 压缩 数据 。 


10.8.1 用 GZIP 进 行 简单 压缩 





GZIP 接 口 非常 简单 ， 所 以 如 果 只 有 单个 数据 流 需 要 压缩 〈 而 不 是 一 系 
列 不 同 的 数据 ) ， 那 么 它 就 可 能 是 最 适当 选择 。 下 面 是 对 单个 文件 进行 
压缩 的 例子 : 


//: GZIPcompress.java 


// Uses Java 1.1 GZIP compression to compress 
// a file whose name is passed on the command 
// line. 
import java.io.*; 
import java.util.zip.*; 
public class GZIPcompress { 
public static void main(String[] args) { 
try { 
BufferedReader in = 
new BufferedReader ( 
new FileReader(args[0])); 
BufferedOutputStream out = 
new BufferedOutputStream( 
new GZIPOutputStream( 
new FileOutputStream("test.gz"))); 
System.out.printin("Writing file"); 
int C; 
while((c = in.read()) != -1) 


out.write(c); 


in.close(); 
out.close(); 
System.out.println("Reading file"); 
BufferedReader in2 = 
new BufferedReader ( 
new InputStreamReader ( 
new GZIPInputStream( 
new FileInputStream("test.gz")))); 
String s; 
while((s = in2.readLine()) != null) 


System.out.println(s); 


WY 


catch(Exception e) { 


e.printStackTrace(); 


} 
ke 


压缩 类 的 用 法 非常 直观 只 需 将 输出 流 封 装 到 一 个 GZIPOutputStream 
或 者 ZipOutputStream 内 ， 并 将 输入 流 封装 到 GZIPInputStream 或 者 
ZipInputStream 内 即 可 。 剩 余 的 全 部 操作 就 是 标准 的 IO 读 写 。 然 而 ， 这 
是 一 个 很 典型 的 例子 ， 我 们 不 得 不 混合 使 用 新 旧 IO 流 : 数据 的 输入 使 用 
Reader 类 ， 而 GZIPOutputStream 的 构建 器 只 能 接收 一 个 OutputStream 对 
象 ， 不 能 接收 Writer 对 象 。 


10.8.2 用 Zip 进 行 多 文件 保存 
提供 了 Zip 支 持 的 Java 1.1 库 显得 更 加 人 全面。 利用 它 可 以 方便 地 保存 多 个 





文件 。 甚 至 有 一 个 独立 的 类 来 简化 对 Zip 文 件 的 读 操作 。 这 个 库 采 采用 
的 是 标准 Zip 格 式 ， 所 以 能 与 当前 因特网 上 使 用 的 大 量 压 缩 、 解 压 工 具 
很 好 地 协作 。 下 面 这 个 例子 采取 了 与 前 例 相 同 的 形式 ， 但 能 根据 我 们 需 
要 控制 任意 数量 的 命令 行 参数 。 除 此 之 外 ， 它 展示 了 如 何 用 Checksum 
类 来 计算 和 校 验 文件 的 “ 校 验 和 ”(Checksum) 。 可 选用 两 种 类 型 的 
Checksum: Adler32 〈 速 度 要 快 一 些 ) 和 CRC32〔( 慢 一 些 ， 但 更 准 

确 ) 。 





//: ZipCompress.java 


// Uses Java 1.1 Zip compression to compress 
// any number of files whose names are passed 
// on the command line. 
import java.io.*; 
import java.util.*; 
import java.util.zip.*; 
public class ZipCompress { 
public static void main(String[] args) { 
try { 
FileOutputStream f = 
new FileOutputStream("test.zip"); 
CheckedOutputStream csum = 
new CheckedOutputStream( 
f, new Adler32()); 
ZipOutputStream out = 


new ZipOutputStream( 


new BufferedOutputStream(csum) ); 
out.setComment("A test of Java Zipping"); 
// Can't read the above comment, though 
for(int i = 0; i < args.length; i++) { 
System.out.printin( 
"Writing file " + args[i]); 
BufferedReader in = 
new BufferedReader ( 
new FileReader(args[i])); 
out.putNextEntry(new ZipEntry(args[i])); 
int c; 
while((c = in.read()) != -1) 
out.write(c); 
in.close(); 
} 
out.close(); 
// Checksum valid only after the file 
// has been closed! 
System.out.println("Checksum: " + 
csum.getChecksum().getValue()); 
// Now extract the files: 
System.out.println("Reading file"); 


FileInputStream fi = 


new FileInputStream("test.zip"); 
CheckedInputStream csumi = 

new CheckedInputStream( 

fi, new Adler32()); 

ZipInputStream in2 = 

new ZipInputStream( 

new BufferedInputStream(csum1) ); 

ZipEntry ze; 
System.out.printin( "Checksum: " + 

csumi.getChecksum().getValue()); 
while((ze = in2.getNextEntry()) != null) { 

System.out.printin("Reading file " + ze); 

int x; 

while((x = in2.read()) != -1) 

System.out.write(x); 
} 
in2.close(); 
// Alternative way to open and read 
// zip files: 

ZipFile zf = new ZipFile("test.zip"); 
Enumeration e = zf.entries(); 
while(e.hasMoreElements()) { 


ZipEntry ze2 = (ZipEntry)e.nextElement(); 


System.out.printin("File: " + ze2); 
// ... and extract the data as before 


} 
} catch(Exception e) { 


e.printStackTrace(); 


} 
SA 


对 于 要 加 入 压缩 档 的 每 一 个 文件 ， 都 必须 调用 putNextEntry0， 并 将 其 传 
递 给 一 个 ZipEntry 对 象 。ZipEntry 对 象 包 含 了 一 个 功能 全 面 的 接口 ， 利 
用 它 可 以 获取 和 设置 Zip 文 件 内 那个 特定 的 Entry AO) 上 能 够 接受 的 
所 有 数据 : 名 字 、 压 缩 后 和 压缩 前 的 长 度 、 日 期 、CRC 校 验 和 、 人 额外 字 
段 的 数据 、 注 释 、 压 缩 方法 以 及 它 是 否 一 个 目录 入 口 等 等 。 然 而 ， 虽 然 
Zip 格 式 提 供 了 设置 密码 的 方法 ， 但 Java 的 Zip 库 没有 提供 这 方面 的 文 
持 。 而 且 尽 管 CheckedInputStream 和 CheckedOutputStream 同 时 提供 了 对 
Adler32 和 CRC32 校 验 和 的 支持 ， 但 是 ZipEntry 只 支持 CRC 的 接口 。 这 虽 
然 属 于 基层 Zip 格 式 的 限制 ， 但 却 限制 了 我 们 使 用 速度 更 快 的 Adler32。 


为 解压 文件 ，ZipInputStream 提 供 了 一 个 getNextEntry() 方 法 ， 能 在 有 的 
前 提 下 返回 下 一 个 ZipEntry。 作 为 一 个 更 简洁 的 方法 ， 可 以 用 ZipFile 对 
象 读 取 文 件 。 访 对象 有 一 个 entries0) 方 法 ， 可 以 为 ZipEntry 返 回 一 个 
Enumeration 〈 枚 举 ) 。 


为 读 取 校 验 和 ， 必 须 多 少 拥 有 对 关联 的 Checksum 对 象 的 访问 权限 。 在 
这 里 保留 了 指 问 CheckedOutputStream 和 CheckedInputStream 对 象 的 一 个 
句 顶 。 但 是 ， 也 可 以 只 占有 指 疝 Checksum 对 象 的 一 个 句 椭 。 


Zip 流 中 一 个 令 人 困惑 的 方法 是 setComment()。 正 如 前 面 展 示 的 那样 ， 我 

们 可 在 写 一 个 文件 时 设置 注释 内 容 ， 但 却 没 有 办 法 取出 ZipInputStream 

看 起 来 ， 似 乎 只 能 通过 ZipEntry 逐 个 入 口 地 提供 对 注释 的 完 
Fo 





当然 ， 使 用 GZIP 或 Zip 库 时 并 不 仅仅 限于 文件 一 一 可 以 压缩 任何 东西 ， 


包括 要 通过 网 络 连接 发 送 的 数据 。 
10.8.3 Java 归 档 (jar) 实用 程序 


Zip 格 式 亦 在 Java 1.1 的 JAR (Java ARchive) 文件 格式 中 得 到 了 采用 。 这 
种 文件 格式 的 作用 是 将 一 系列 文件 合并 到 单个 压缩 文件 里 ， 就 象 Zip 那 

样 。 然 而 ， 同 Java 中 其 他 任何 东西 一 样 ，JAR 文 件 是 跨 平 台 的 ， 所 以 不 

必 关 心 涉 及 具体 平台 的 问题 。 除 了 可 以 包括 声音 和 图 像 文 件 以 外 ， 也 可 
以 在 其 中 包括 类 文件 。 


涉及 因特网 应 用 时 ，JAR 文 件 显 得 特别 有 用 。 在 JAR 文 件 之 前 ，Web 浏 
览 右 必须 重复 多 次 请 求 Web 服 务 器 ， 以 便 下 载 完 构成 一 个 “程序 

F” CApplet) 的 所 有 文件 。 除 此 以 外 ， 每 个 文件 都 是 未 经 压缩 的 。 但 在 
将 所 有 这 些 文件 合并 到 一 个 JAR 文 件 里 以 后 ， 只 需 同 远程 服务 器 发 出 一 
次 请 求 即 可 。 同 时 ， 由 于 采用 了 压缩 技术 ， 所 以 可 在 更 短 的 时 间 里 获得 
全 部 数据 。 另 外 ，JAR 文 件 里 的 每 个 入 口 〈 和 条目) 都 可 以 加 上 数字 化 签 
名 【详情 参考 Java 用 户 文档 ) 。 

一 个 JAR 文 件 由 一 系列 采用 Zip 压 缩 格 式 的 文件 构成 ， 同 时 还 有 一 张 “ 详 
情 单 ”， 对 所 有 这 些 文件 进行 了 摘 述 《可 创建 自己 的 详情 单 文件 ， 合 
则 ，jar 程 序 会 为 我 们 代劳 ) 。 在 联机 用 户 文档 中 ， 可 以 找到 与 JAR 详 情 
单 更 多 的 资料 (详情 单 的 英语 是 “Manifest”) 。 


jar 实 用 程序 已 与 Sun 的 JDK 配 套 提供 ， 可 以 按 我 们 的 选择 自动 压缩 文 
件 。 请 在 命令 行 调用 它 : 


jar [选项 ] 说 明 [详情 单 ] 输入 文件 


其 中 , “选项 ”用 一 系列 字母 表示 不 必 输 入 连 字 写 或 其 他 任何 指示 
FF) 。 如 下 所 示 : 


c 创 建新 的 或 空 的 压缩 档 
t 列 出 目录 表 

x 解压 所 有 文件 

x file 解压 指定 文件 

















f 指出 “我 准备 同 你 提供 文件 名 >”。 知 省 略 此 参数 ，jar 会 假定 它 的 输入 来 
目标 准 输入 ; 或 者 在 它 创建 文件 时 ， 输 出 会 进入 标准 输出 内 


m 指出 第 一 个 参数 将 是 用 户 目 建 的 详情 表 文 件 的 名 字 
Y 产 生 详细 输出 ， 对 jar 做 的 工作 进行 巨细 无 中 的 描述 


O 只 保存 文件 ; 不 压缩 文件 (用 于 创建 一 个 JAR 文 件 ， 以 便 我 们 将 其 置 
入 目 己 的 类 路 径 中 ) 


M 不 自动 生成 详情 表 文 件 

在 准备 进入 JAR 文 件 的 文件 中 ， 寿 包括 了 一 个 子 目录 ， 那 个 子 目录 会 自 
>” 其 中 包括 它 上 自己 的 所 有 子 目录 ， 以 此 类 推 。 路 径 信息 也 会 得 到 
AEH 。 

下 面 是 调用 jar 的 一 些 典 型 方法 : 





jar cf myJarFile.jar *.class 


用 于 创建 一 个 名 为 myjJarFile.jar 的 JAR 文 件 ， 其 中 包含 了 当前 目录 中 的 所 
有 类 文件 ， 同 时 还 有 自动 产生 的 详情 表 文件 。 








jar cmf myJarFile.jar myManifestFile.mf *.class 


与 前 例 类 似 ， 但 添加 了 一 个 名 为 myManifestFile.mf 的 用 户 自 建 详情 表 文 
件 。 


jar tf myJarFile.jar 
生成 myJarFile.jar 内 所 有 文件 的 一 个 目录 表 。 
jar tvf myJarFile.jar 


添加 “verbose”( 详 尽 ) 标志 ， 提 供与 nyJarFile.jar 中 的 文件 有 关 的 、 更 
详细 的 资料 。 


jar cvf myApp.jar audio classes image 





假定 audio，classes 和 image 是 子 目 录 ， 这 样 便 将 所 有 子 目 录 合 并 到 文件 
eae 。 其 中 也 包括 了 “verbose” 标 志 ， 可 在 jar 程 序 工 作 时 反馈 
详尽 的 信息 。 


如 果 用 O 选 项 创建 了 一 个 JAR 文 件 ， 那 个 文件 束 可 置 入 目 己 的 类 路 径 
(CLASSPATH) 中 : 


CLASSPATH="lib1.jar;lib2.jar;" 

Java 能 在 lib1.jar 和 lib2.jar 中 搜索 目标 类 文件 。 

jar 工 具 的 功能 没有 zip 工 具 那 么 丰富 。 例 如 ， 不 能 够 添加 或 更 新 一 个 现 
成 JAR 文 件 中 的 文件 ， 只 能 从 头 开始 新 建 一 个 JAR 文 件 。 此 外 ， 不 能 将 
文件 移入 一 个 JAR 文 件 ， 并 在 移动 后 将 它们 删除 。 然 而 ， 在 一 种 平台 上 
创建 的 JAR 文 件 可 在 其 他 任何 平台 上 由 jar 工 具 守 无 阻碍 地 读 出 《〈 这 个 问 
题 有 时 会 困扰 zip 工 具 ) 。 


正如 大 家 在 第 13 章 会 看 到 的 那样 ， 我 们 也 用 JAR 为 Java Beans 打 包 。 


10.9 对 象 序 列 化 


Java 1.1 增 添 了 一 种 有 趣 的 特性 ， 名 为 “对 象 序列 化 ”(Object 
Serialization) 。 它 面向 那些 实现 了 Serializable 接 口 的 对 象 ， 可 将 它们 转 
换 成 一 系列 字 节 ， 并 可 在 以 后 完全 恢复 回 原 来 的 样子 。 这 一 过 程 亦 可 通 
过 网 络 进行 。 这 意味 着 序列 化 机 制 能 目 动 补偿 操作 系统 间 的 差异 。 换 名 
话说 ， 可 以 先 在 Windows 机 器 上 创建 一 个 对 象 ， 对 其 序列 化 ， 然 后 通过 
网 络 发 给 一 台 Unix 机 右 ， 然 后 在 那里 准确 无 误 地 重新 “装配 ”。 不 必 关 心 
数据 在 个 同 机 器 上 如 何 表 未， 也 不 必 关 心 字 节 的 顺序 或 者 其 他 任何 细 
ae 














就 其 本 身 来 说 ， 对 象 的 序列 化 是 非常 有 趣 的 ， 因 为 利用 它 可 以 实现 “有 
限 持 久 化 ”。 请 记 住 “持久 化 ”意味 着 对 象 的 “生存 时 间 ” 并 不 取决 于 程序 
是 否 正在 执行 一 一 它 存在 或 “生存 ”于 程序 的 每 一 次 调用 之 间 。 通 过 序列 
化 一 个 对 象 ， 将 其 写 入 磁盘 ， 以 后 在 程序 重新 调用 时 重新 恢复 那个 对 
象 ， 就 能 圆满 实现 一 种 “持久 ”效果 。 之 所 以 称 其 为 “有限 ”， 有 是 因为 不 能 
用 某 种 “persistent”( 持 久 〉 关键 字 简单 地 地 定义 一 个 对 象 ， 并 让 系统 目 
动 照看 其 他 所 有 细 贡 问题 《尽管 将 来 可 能 成 为 现实 ) 。 相 反 ， 必 须 在 目 
己 的 程序 中 明确 地 序列 化 和 组 装 对 象 。 


语言 里 增加 了 对 象 序 列 化 的 概念 后 ， 可 提供 对 两 种 主要 特性 的 支持 。 
Java 1.1 的 “远程 方法 调用 ”(RMI) 使 本 来 存在 于 其 他 机 器 的 对 象 可 以 表 
现 出 好 象 就 在 本 地 机 器 上 的 行为 。 将 消息 发 给 远程 对 象 时 ， 需 要 通过 对 
象 序列 化 来 传输 参数 和 返回 值 。RMI 将 在 第 15 章 作 有 具体 讨论 。 

对 象 的 序列 化 也 是 Java ”Beans 必需 的 ， 后 者 由 Java _ 1.1 引入。 使 用 一 个 


Bean 时 ， 它 的 状态 信息 通常 在 设计 期 间 配 置 好 。 程 序 局 动 以 后 ， 这 种 状 
态 信息 必须 保存 下 来 ， 以 便 程 序 局 动 以 后 恢复 ， 其 体 工 作 由 对 象 序列 化 
































完成 


对 象 的 序列 化 处 理 非常 简单 ， 只 需 对 象 实 现 了 Serializable 接 口 即 可 (该 
接口 仅 是 一 个 标记 ， 没 有 方法 ) 。 在 Java ”1.1 中 ， 许 多 标准 库 类 都 发 生 
了 改变 ， 以 便 能 够 序列 化 一 一 其 中 包括 用 于 基本 数据 类 型 的 全 部 封装 
器 、 所 有 集合 类 以 及 其 他 许多 东西 。 甚 至 Class 对 象 也 可 以 序列 化 (第 11 
章 讲述 了 有 具体 实现 过 程 ) 。 


为 序列 化 一 个 对 象 ， 首 先 要 创建 菜 些 OutputStream 对 象 ， 然 后 将 其 封装 
到 ObjectOutputStream 对 象 内 。 此 时 ， 只 需 调 用 writeObjectO 即 可 完成 对 
象 的 序列 化 ， 并 将 其 发 送 给 OutputStream。 相 反 的 过 程 是 将 一 个 
InputStream 封 装 到 ObjectInputStream 内 ， 然 后 调用 readObject0。 和 往 季 
一 样 ， 我 们 最 后 获得 的 是 指 癌 一 个 上 漳 造 型 Object 的 句柄 ， 所 以 必须 下 
渊 造型， 以便 能 够 直接 设置 。 


对 象 序列 化 特别 “聪明 ”的 一 个 地 方 是 它 不 仅 保 存 了 对 象 的 "全景 图 ”， 而 
且 能 退 踪 对 象 内 包含 的 所 有 句柄 并 保存 那些 对 象 ， 接 着 又 能 对 每 个 对 象 
内 包含 的 句柄 进行 追踪 以 此 类 推 。 我 们 有 时 将 这 种 情况 称 为 “对象 
网 ”>， 单 个 对 象 可 与 之 建立 连接 。 而 且 它 还 包含 了 对 象 的 句柄 数组 以 及 
成 员 对 象 。 知 必须 目 行 操 纵 一 套 对 象 序列 化 机 制 ， 那 么 在 代码 里 追 踩 所 
有 这 些 链 接 时 可 能 会 显得 非常 麻烦 。 在 另 一 方面 ， 由 于 Java 对 象 的 序列 
化 似乎 找 不 出 什么 缺点 ， 所 以 请 尽量 不 要 目 己 动手 ， 让 它 用 优化 的 算法 
目 动 维护 整个 对 象 网 。 下 面 这 个 例子 对 序列 化 机 制 进行 了 测试 。 它 建立 
丁 许多 链接 对 象 的 一 个 “Worm”( 蜂 忠 ) ， 每 个 对 象 都 与 Vorm 中 的 下 一 
段 链接 ， 同 时 又 与 属于 不 同类 (Data) 的 对 象 句柄 数组 链接 : 


//: Worm,java 





























// Demonstrates object serialization in Java 1.1 
import java.io.*; 
class Data implements Serializable { 

private int i; 

Data(int x) { i = x; } 

public String toString() { 


return Integer.toString(1i); 


} 


public class Worm implements Serializable { 


// Generate a random int value: 
private static int r() { 

return (int)(Math.random() * 10); 
} 
private Data[] d = { 

new Data(r()), new Data(r()), new Data(r()) 
}; 
private Worm next; 
private char c; 
// Value of i == number of segments 
Worm(int i, char x) { 


System.out.printin(" Worm constructor: " + i); 


if(--i > 0) 


next = new Worm(i, (char)(x + 1)); 


Worm() { 

System.out.println("Default constructor"); 
5 
public String toString() { 

String s = ":" + c + "("; 

for(int i = 0; i < d.length; i++) 


s += d[i].toString(); 


if(next != null) 
s += next.toString(); 
return s; 
} 
public static void main(String[] args) { 
Worm w = new Worm(6, ‘a'); 
System.out.printin("w = " + w); 
try { 
ObjectOutputStream out = 
new ObjectOutputStream( 
new FileOutputStream("worm.out")); 
out.writeObject("Worm storage"); 
out.writeObject(w); 
out.close(); // Also flushes output 
ObjectInputStream in = 
new ObjectInputStream( 
new FileInputStream("worm.out")); 
String s = (String)in.readObject(); 
Worm w2 = (Worm)in.readObject(); 
System.out.println(s + ", w2 = " + w2); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
try { 
ByteArrayOutputStream bout = 
new ByteArrayOutputStream(); 
ObjectOutputStream out = 
new ObjectOutputStream(bout ) ， 
out.writeObject("Worm storage"); 
out.writeObject(w); 
out.flush(); 
ObjectInputStream in = 
new ObjectInputStream( 
new ByteArrayInputStream( 
bout.toByteArray())); 
String s = (String)in.readObject(); 
Worm w3 = (Worm)in.readObject(); 
System.out.println(s + ", w3 = " + w3); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
} S/S: 


l 


更 有 趣 的 是 ，Worm 内 的 Data 对 象 数组 是 用 随机 数字 初始 化 的 (这 样 便 





不 用 怀疑 编译 器 保留 了 某 种 原始 信息 ) 。 每 个 Worm 段 都 用 一 个 Char 标 
记 。 这 个 Char 是 在 重复 生成 链接 的 Worm 列 表 时 自动 产生 的 。 创 建 一 个 
Worm 时 ， 需 告诉 构建 器 希望 它 有 多 长 。 为 产生 下 一 个 句柄 (next) ， 
它 总 是 用 减 去 1 的 长 度 来 调用 Worm 构 建 器 。 最 后 一 个 next 句 柄 则 保持 为 
null (F) ， 表 示 已 抵达 Worm 的 尾部 。 


上 面 的 所 有 操作 都 是 为 了 加 深 事 情 的 复杂 程度 ， 加 大 对 象 序列 化 的 难 
度 。 然 而 ， 真 正 的 序列 化 过 程 却 是 非常 简单 的 。 一 旦 从 另外 某 个 流 里 创 
建 了 ObjectOutputStream，writeObjectO 就 会 序列 化 对 象 。 注 意 也 可 以 为 
一 个 String 调 用 writeObject()。 亦 可 使 用 与 DataOutputStream 相 同 的 方法 
写 入 所 有 基本 数据 类 型 (它们 有 相同 的 接口 )。 


有 两 个 单独 的 try 块 看 起 来 是 类 似 的 。 第 一 个 读 写 的 是 文件 ， 而 男 一 个 读 
写 的 是 一 个 ByteArray〈 字 节 数 组 ) 。 可 利用 对 任何 DataInputStream 或 者 
DataOutputStream 的 序列 化 来 读 写 特定 的 对 象 ， 正 如 在 关于 连 网 的 那 一 
章 会 讲 到 的 那样 ， 这 些 对 象 甚至 包括 网 络 。 一 次 循环 后 的 输出 结果 如 
































Worm constructor: 6 


Worm constructor: 5 

Worm constructor: 4 

Worm constructor: 3 

Worm constructor: 2 

Worm constructor: 1 

w = :a(262):b(100) :c(396) :d(480) :e(316) :f(398) 

Worm storage, w2 = :a(262):b(100):c(396) :d(480) :e(316) :f(398) 


Worm storage, w3 = :a(262):b(100):c(396) :d(480) :e(316) :f(398) 


ae > BACT] TRAIT ASE LT SORA OT Fe BE 


注意 在 对 一 个 Serializable〈 可 序列 化 ) 对 象 进行 重 新 装配 的 过 程 中 ， 不 
会 调用 任何 构建 问 《〈 甚 至 默认 构建 器 ) 。 整 个 对 象 都 是 通过 从 
InputStream 中 取得 数据 恢复 的 。 


作为 Java 1.1 特 性 的 一 种 ， 我 们 注意 到 对 象 的 序列 化 并 不 属于 新 的 Reader 
和 Writer 层 次 结构 的 一 部 分 ， 而 是 沿用 老式 的 InputStream 和 OutputStream 
‘te 所 以 在 一 些 特殊 的 场合 下 ， 不 得 不 混合 使 用 两 种 类 型 的 层次 结 
10.9.1 寻找 类 

读者 或 许 会 奇怪 为 什么 需要 一 个 对 象 从 它 的 序列 化 状态 中 恢复 。 举 个 例 
子 来 说 ， 假 定 我 们 序列 化 一 个 对 象 ， 并 通过 网 络 将 其 作为 文件 传送 给 另 
一 台 机 器 。 此 时 ， 位 于 另 一 台 机 器 的 程序 可 以 只 用 文件 目录 来 重新 构造 
这 个 对 象 吗 ? 


回答 这 个 问题 的 最 好 方法 就 是 做 一 个 实验 。 下 面 这 个 文件 位 于 本 章 的 子 
目录 下 ; 


//: Alien.java 























// A serializable class 
import java.io.*; 
public class Alien implements Serializable { 


Plt i= 





用 于 创建 和 序列 化 一 个 Alien 对 象 的 文件 位 于 相同 的 目录 下 : 


//: FreezeAlien.java 


// Create a serialized output file 
import java.io.*; 


public class FreezeAlien { 


public static void main(String[] args) 
throws Exception { 
ObjectOutput out = 
new ObjectOutputStream( 
new FileOutputStream("file.x")); 
Alien zorcon = new Alien(); 
out.writeObject(zorcon); 


} 
SS 


该 程序 并 不 是 捕获 和 控制 违例 ， 而 是 将 违例 人 简单、 直接 地 传递 到 main() 
外 部 ， 这 样 便 能 在 命令 行 报告 它们 。 

a KRT E MN file. x32 hil 3 4 Axfilesi + Aa, AR 
码 如 下 : 


//: ThawAlien.java 


// Try to recover a serialized file without the 
// class of object that's stored in that file. 
package ci0.xfiles; 
import java.io.*; 
public class ThawAlien { 
public static void main(String[] args) 
throws Exception { 


ObjectInputStream in = 


new ObjectInputStream( 
new FileInputStream("file.x")); 
Object mystery = in.readObject(); 
System. out.printin( 
mystery.getClass().toString()); 


} 
} ///:~ 


该 程序 能 打开 文件 ， 并 成 功 读 取 mystery 对 象 中 的 内 容 。 然 而 ， 一 旦 党 
试 查 找 与 对 象 有 关 的 任何 资料 这 要 求 Alien 的 Class 对 象 Java hie 
机 (JVM) 便 找 不 到 Alien.class〈 除 非 它 正好 在 类 路 径 内 ， 而 本 例 理 应 
相反 ) 。 这 样 就 会 得 到 一 个 名 叫 ClassNotFoundException 的 违例 (同样 
地 ， 知 非 能 够 校 验 Alien 存 在 的 证 据 ， 否 则 它 等 于 消失 ) 。 


恢复 了 一 个 序列 化 的 对 象 后 ， 如 果 想 对 其 做 更 多 的 事情 ， 必 须 保证 JVM 
能 在 本 地 类 路 径 或 者 因特网 的 其 他 什么 地 方 找到 相关 的 .class 文 件 。 


10.9.2 序列 化 的 控制 


正如 大 家 看 到 的 那样 ， 默 认 的 序列 化 机 制 并 不 难 操纵 。 然 而 ， 假 行 有 特 
殊 要 求 叉 该 怎么 办 呢 ? 我 们 可 能 有 特殊 的 安全 问题 ， 不 希望 对 象 的 菏 一 
部 分 序列 化 ， 或 者 东 一 个 子 对 象 完 全 不 必 序 列 化 ， 因 为 对 象 恢复 以 后 ， 
那 一 部 分 需要 重新 创建 。 


此 时 ， 通 过 实现 Externalizable 接 口 ， 用 它 人 代替 Serializable 接 口 ， 便 可 控 
制 序列 化 的 具体 过 程 。 这 个 Externalizable 接 口 扩展 了 Serializable， 并 增 
添 了 两 个 方法 : writeExternal() 和 readExternal()。 在 序列 化 和 重新 装配 的 
过 程 中 ， 会 自动 调用 这 两 个 方法 ， 以 便 我 们 执行 一 些 特殊 操作 。 

下 面 这 个 例子 展示 了 Externalizable 接 口 方法 的 简单 应 用 。 注 意 Blip1 和 
ee A ea 除了 极 微 小 的 差别 (自己 研究 一 下 代码 ， 看 看 是 
PAE ACH): 


//: Blips.java 

















// Simple use of Externalizable & a pitfall 
import java.io.*; 
import java.util.*; 
class Blip1 implements Externalizable { 
public Blipi() { 
System.out.printin("Blip1 Constructor"); 
} 
public void writeExternal(ObjectOutput out) 
throws IOException { 
System.out.printiln("Blip1.writeExternal"); 
} 
public void readExternal(ObjectInput in) 
throws IOException, ClassNotFoundException { 


System.out.printin("Blip1.readExternal"); 


} 


class Blip2 implements Externalizable { 
Blip2() { 
System.out.printin("Blip2 Constructor"); 
} 
public void writeExternal(ObjectOutput out) 


throws IOException { 


System.out.printin("Blip2.writeExternal"); 
} 
public void readExternal(ObjectInput in) 
throws IOException, ClassNotFoundException { 


System.out.printin("Blip2.readExternal"); 


} 


public class Blips { 
public static void main(String[] args) { 
System.out.println("Constructing objects:"); 


Blip1 b1 = new Blip1(); 


Blip2 b2 = new Blip2(); 
try { 
ObjectOutputStream o = 
new ObjectOutputStream( 
new FileOutputStream("Blips.out")); 
System.out.println( "Saving objects:"); 
o.writeObject(b1); 
o.writeObject(b2); 
o.close(); 
// Now get them back: 
ObjectInputStream in = 


new ObjectInputStream( 


new FileInputStream("Blips.out")); 
System.out.println( "Recovering bi:"); 
b1 = (Blip1)in.readObject(); 
// OOPS! Throws an exception: 
//! System.out.println("Recovering b2:"); 
//! b2 = (Blip2)in.readObject(); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
} ///:~ 


该 程序 输出 如 下 : 


Constructing objects : 


BJip1 Constructor 
Blip2 Constructor 
Saving objects: 
Blipi.writeExternal 
Blip2.writeExternal 
Recovering b1: 
Blip1i Constructor 


Blip1.readExternal 


未 恢复 Blip2 对 象 的 原因 是 那样 做 会 导致 一 个 违例 。 你 找 出 了 Blip1 和 
Blip2 之 间 的 区 别 吗 ? Blip1 的 构建 器 是 “公共 的 ”(public) ，Blip2 的 构建 
器 则 不 然 ， 这 样 便 会 在 恢复 时 造成 违例 。 试 试 将 Blip2 的 构建 器 属性 变 
成 “public”， 然 后 删除 ! 注 释 标 记 ， 看 看 是 售 能 得 到 正确 的 结 


恢复 b1 后 ， 会 调用 Blip1 默 认 构建 器 。 这 与 恢复 一 个 Serializable〈 可 序列 
化 ) 对 象 不 同 。 在 后 者 的 情况 下 ， 对 象 完全 以 它 保存 下 来 的 二 进 制 位 为 
基础 恢复 ， 不 存在 构建 器 调用 。 而 对 一 个 Externalizable 对 象 ， 所 有 普通 
的 默认 构建 行为 都 会 发 生 《〈 包 括 在 字段 定义 时 的 初始 化 ) ， 而 且 会 调用 
readExternal()。 必 须 注意 这 一 事实 特别 注意 所 有 默认 的 构建 行为 都 
会 进行 否则 很 难 在 自己 的 Externalizable 对 象 中 产生 正确 的 行为 。 


a 例子 揭示 了 保存 和 恢复 一 个 Externalizable 对 象 必须 做 的 全 部 事 
情 : 


























//: Blip3.java 


// Reconstructing an externalizable object 
import java.io.*; 
import java.util.*; 
class Blip3 implements Externalizable { 
int 1; 
String s; // No initialization 
public Blip3() { 
System.out.printin("Blip3 Constructor"); 
// s, i not initialized 
} 
public Blip3(String x, int a) { 


System.out.printin("Blip3(String x, int a)"); 


S = xX} 
i = a; 
// s & i initialized only in non-default 
// constructor. 

} 

public String toString() { return s + i; } 

public void writeExternal(ObjectOutput out) 

throws IOException { 

System.out.printin("Blip3.writeExternal"); 
// You must do this: 
out.writeObject(s); out.writeInt(i); 

} 

public void readExternal(ObjectInput in) 
throws IOException, ClassNotFoundException { 
System.out.printin("Blip3.readExternal"); 
// You must do this: 
s = (String)in.readObject(); 
i =in.readInt(); 

} 

public static void main(String[] args) { 
System.out.println("Constructing objects:"); 
Blip3 b3 = new Blip3("A String ", 47); 


System.out.printin(b3.toString()); 


try { 
ObjectOutputStream o = 
new ObjectOutputStream( 
new FileOutputStream("Blip3.out")); 
System.out.println( "Saving object:"); 
o.writeObject(b3); 
o.close(); 
// Now get it back: 
ObjectInputStream in = 
new ObjectInputStream( 
new FileInputStream("Blip3.out")); 
System.out.println( "Recovering b3:"); 
b3 = (Blip3)in.readObject(); 
System.out.println(b3.toString()); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
} MAS i~ 


其 中 ， 字 段 s 和 ij 只 在 第 二 个 构建 器 中 初始 化 ， 不 关 默 认 构 建 嚣 的 事 。 这 
意味 着 假如 不 在 readExternal 中 初始 化 s: 和 i， 它 们 就 会 成 为 null (因为 在 

对 象 创建 的 第 一 步 中 已 将 对 象 的 存储 空间 清除 为 1) o AER HRE 

于 “You must do this” 后 面 的 两 行 代 码 ， 并 运行 程序 ， 就 会 发 现 当 对 象 恢 
复 以 后 ，s 是 null， 而 i 是 零 。 











看 从 一 个 Externalizable 对 象 继承 ， 通 和 需要 调用 writeExternalO0 和 
readExternal() 的 基础 类 版 本 ， 以 便 正确 地 保存 和 恢复 基础 类 组 件 。 


所 以 为 了 让 一 切 正常 运作 起 来 ， 干 万 不 可 仪 在 writeExternal() 方 法 执行 期 
间 写 入 对 象 的 重要 数据 (没有 默认 的 行为 可 用 来 为 一 个 Externalizable 对 
象 写 入 所 有 成 员 对 象 ) 的 ， 而 是 必须 在 readExternal0) 方 法 中 也 恢复 那些 
数据 。 初 次 操作 时 可 能 会 有 些 不 习惯 ， 因 为 Externalizable 对 象 的 默认 构 
和 但 实情 并 非 如 








1. transient (临时 〉 关键 字 


控制 序列 化 过 程 时 ， 可 能 有 一 个 特定 的 子 对 象 不 愿 让 Java 的 序列 化 机 制 
自动 保存 与 恢复 。 一 般 地 ， 大 那个 子 对 象 包含 了 不 想 序 列 化 的 敏感 信息 
《如 密码 ) ， 就 会 面临 这 种 情况 。 即 使 那 种 信息 在 对 象 中 具 

有 “private”( 私 有 ) 属性 ， 但 一 旦 经 序列 化 处 理 ， 人 们 就 可 以 通过 读 取 
一 个 文件 ， 或 者 拦截 网 络 传输 得 到 它 。 


为 防止 对 象 的 敏感 部 分 被 序列 化 ， 一 个 办 法 是 将 自己 的 类 实现 为 
Externalizable， 束 象 前 面 展 示 的 那样 。 这 样 一 来 ， 没 有 任何 东西 可 以 自 
动 序列 化 ， 只 能 在 writeExternalO 明 确 序列 化 那些 需要 的 部 分 。 


然而 ， 若 操作 的 是 一 个 Serializable 对 象 ， 所 有 序列 化 操作 都 会 自动 进 
行 。 为 解决 这 个 问题 ， 可 以 用 transient (临时 ) 逐个 字段 地 关闭 序列 
化 ， 它 的 意思 是 “不 要 麻烦 你 〈 指 自动 机 制 ) 保存 或 恢复 它 了 一 一 我 会 
自己 处 理 的 ”。 

例如 ， 假 设 一 个 Login 对 象 包含 了 与 一 个 特定 的 登录 会 话 有 关 的 信息 。 
校 验 登 录 的 合法 性 时 ， 一 般 都 想 将 数据 保存 下 来 ， 但 不 包括 密码 。 为 做 
到 这 一 点 ， 最 简单 的 办 法 是 实现 Serializable， 并 将 password 字 段 设 为 
transient。 下 面 是 具体 的 代码 : 


//: Logon.java 





























// Demonstrates the "transient" keyword 


import java.io.*; 


import java.util.*; 
class Logon implements Serializable { 
private Date date = new Date(); 
private String username; 
private transient String password; 
Logon(String name, String pwd) { 
username = name; 


password = pwd; 


} 


public String toString() { 


String pwd = 

(password == null) ? "(n/a)" : password; 
return "logon info: \n Nes 

"username: " + username + 


"\n date: " + date.toString() + 
"\n password: " + pwd; 
} 
public static void main(String[] args) { 
Logon a = new Logon("Hulk", "myLittlePony") ; 
System.out.println( "logon a = " + a); 
try { 
ObjectOutputStream o = 


new ObjectOutputStream( 


new FileOutputStream("Logon.out")); 
o.writeObject(a); 
o.close(); 
// Delay: 
int seconds = 5; 
long t = System.currentTimeMillis() 

+ seconds * 1000; 
while(System.currentTimeMillis() < t) 
// Now get them back: 

ObjectInputStream in = 

new ObjectInputStream( 

new FileInputStream("Logon.out")); 

System. out.printin( 

"Recovering object at " + new Date()); 
a = (Logon)in.readObject(); 
System.out.println( "logon a = " + a); 

} catch(Exception e) { 


e.printStackTrace(); 


} ///:~ 


可 以 看 到 ， (未 设 成 
transient) ， 所 以 会 目 动 序列 化 。 然 而 ，password 被 设 为 transient， 所 以 
另外 ， 目 动 序列 化 机 制 也 不 会 作 恢复 它 的 尝试。 
俞 出 如 下 : 


logon a = logon info: 


username: Hulk 

date: Sun Mar 23 18:25:53 PST 1997 

password: myLittlePony 
Recovering object at Sun Mar 23 18:25:59 PST 1997 
logon a = logon info: 

username: Hulk 

date: Sun Mar 23 18:25:53 PST 1997 


password: (n/a) 





一 旦 对 象 恢 复 成 原来 的 样子 ，password 字 上段 就 会 变 成 null。 注 意 必须 用 
toString() 检 查 password 是 否 为 null， 因 为 知 用 过 载 的 “+ 运算 符 来 妆 配 一 
个 String 对 象 ， 而 且 那 个 运算 符 直 到 一 个 null 句 柄 ， 束 会 造成 一 个 名 为 
NullPointerException 的 违例 (新 版 Java 可 能 会 提供 避免 这 个 问题 的 代 
AS) is 


我 们 也 发 现 date 字 段 被 保存 到 磁盘 ， 并 从 磁盘 恢复 ， 没 有 重新 生成 。 


由 于 Extemalizable 对 象 默 认 时 不 保存 它 的 任何 字段 ， 所 以 transient 关 键 
字 只 能 伴随 Serializable 使 用 。 


2. Externalizable 的 替代 方法 


若 不 是 特别 在 意 要 实现 Externalizable 接 口 ， 还 有 男 一 种 方法 可 供 选 用 。 
我 们 可 以 实现 Serializable 接 口 ， 并 添加 “〈 注 意 是 “添加 ”， 而 非 “ 履 盖 ? 或 
者 “实现 ”) 名 为 writeObject0 和 readObjectO0 的 方法 。 一 旦 对 象 被 序列 化 





或 者 重新 装配 ， 束 会 分 别 调用 那 两 个 方法 。 也 就 是 说 ， 只 要 提供 了 这 两 
个 方法 ， 就 会 优先 使 用 它们 ， 而 不 考虑 默认 的 序列 化 机 制 。 


这 些 方法 必须 含有 下 列 准 确 的 签名 : 


private void 


writeObject(ObjectOutputStream stream) 
throws IOException; 
private void 
readObject(ObjectInputStream stream) 


throws IOException, ClassNotFoundException 


从 设计 的 角度 出 发 ， 情 况 变 得 有 些 扑 闭 迷离。 首先 ， 大 家 可 能 认为 这 些 
方法 不 属于 基础 类 或 者 Serializable 接 口 的 一 部 分 ， 它 们 应 该 在 自己 的 接 
口中 得 到 定义 。 但 请 注意 它们 被 定义 成 “private”"， 这 和 童 味 着 它们 只 能 由 
这 个 类 的 其 他 成 员 调 用 。 然 而 ， 我 们 实际 并 不 从 这 个 类 的 其 他 成 员 中 调 
用 它们 ， 而 是 由 ObjectOutputStream 和 ObjectInputStream 的 writeObjectO) 
及 readObject() 方 法 来 调用 我 们 对 象 的 writeObject() 和 readObject() 方 法 
(注意 我 在 这 里 用 了 很 大 的 抑制 力 来 避免 使 用 相同 的 方法 名 因为 怕 
WA) 。 大 家 可 能 奇怪 ObjectOutputStream 和 ObjectInputStream 如 何 有 权 
id 问 我 们 的 类 的 private 方 法 一 一 只 能 认为 这 是 序列 化 机 制 玩 的 一 个 把 
戏 。 


在 任何 情况 下 ， 接 口中 的 定义 的 任何 东西 都 会 自动 具有 public 属 性 ， 所 
以 假若 writeObject() 和 readObjectO 必 须 为 private， 那 么 它们 不 能 成 为 接 
O Cinterface) 的 一 部 分 。 但 由 于 我 们 准确 地 加 上 了 签名 ， 所 以 最 终 的 
效果 实际 与 实现 一 个 接口 是 相同 的 。 


看 起 来 似乎 我 们 调用 ObjectOutputStream.writeObjectO 的 时 候 ， 我 们 传递 
给 它 的 Serializable 对 象 似乎 会 被 检查 是 否 实现 了 自己 的 writeObject()。 硅 
答案 是 肯定 的 是 ， 便 会 跳 过 常规 的 序列 化 过 程 ， 并 调用 writeObject()。 
readObject0 也 会 遇 到 同样 的 情况 。 




















还 存在 另 一 个 问题 。 在 我 们 的 writeObjectO 内 部 ， 可 以 调用 
defaultWriteObject()， 从 而 决定 采取 默认 的 writeObject(0 行 动 。 类 似 地 ， 
在 readObject() 内 部 ， 可 以 调用 defaultReadObject()。 下 面 这 个 简单 的 例 
子 演示 了 如 何 对 一 个 Serializable 对 象 的 存储 与 恢复 进行 控制 : 


//: SerialCtl.java 


// Controlling serialization by adding your own 
// writeObject() and readObject() methods. 
import java.io.*; 
public class SerialCtl implements Serializable { 

String a; 

transient String b; 

public SerialCtl(String aa, String bb) { 

a = "Not Transient: " + aa; 


b 


"Transient: " + bb; 

} 

public String toString() { 
return a + "\n" + b; 

} 

private void 
writeObject(ObjectOutputStream stream) 

throws IOException { 

stream.defaultWriteObject(); 


stream.writeObject(b); 


} 


private void 
readObject(ObjectInputStream stream) 
throws IOException, ClassNotFoundException { 
stream.defaultReadObject(); 
b = (String)stream.readObject(); 
} 
public static void main(String[] args) { 
SerialCtl sc = 
new SerialCtl("Testi", "Test2"); 
System.out.printin("Before:\n" + sc); 
ByteArrayOutputStream buf = 
new ByteArrayOutputStream(); 
try { 
ObjectOutputStream o = 
new ObjectOutputStream(buf); 
o.writeObject(sc); 
// Now get it back: 
ObjectInputStream in = 
new ObjectInputStream( 
new ByteArrayInputStream( 
buf.toByteArray())); 


SerialCtl sc2 = (SerialCtl)in.readObject(); 


System.out.printin("After:\n" + sc2); 
} catch(Exception e) { 


e.printStackTrace(); 


} 
} ///:~ 


在 这 个 例子 中 ， 一 个 String 保 持原 始 状态 ， 其 他 设 为 transient〈I 临 时 ) ， 
以 便 证 明 非 临时 字段 会 被 defaultWriteObjectO 方 法 自动 保存 ， 而 transient 
字段 必须 在 程序 中 明确 保存 和 恢复 。 字 段 是 在 构建 器 内 部 初始 化 的 ， 而 
To 时 候 ， 这 证 明了 它们 不 会 在 重新 装配 的 时 候 被 某 些 上 自动 化 
儿 制 初始 化 。 


看 准备 通过 默认 机 制 写 入 对 象 的 非 transient 部 分 ， 那 么 必须 调用 
defaultWriteObject0， 令 其 作为 writeObjectO0 中 的 第 一 个 操作 并 调用 
defaultReadObject0， 令 其 作为 readObjectO 的 第 一 个 操作 。 这 些 都 是 不 
第 见 的 调用 方法 。 举 个 例子 来 次 ， 当 我 们 为 一 个 ObjectOutputStream 调 
用 defaultWriteObjectO 的 时 候 ， 而 且 没 有 为 其 传递 参数 ， 就 需要 采取 这 
种 操作 ， 使 其 知道 对 象 的 句柄 以 及 如 何 写 入 所 有 非 transient 的 部 分 。 这 
种 做 法 非常 不 便 。 


transient 对 象 的 存储 与 恢复 采用 了 我 们 更 熟悉 的 代码 。 现 在 考虑 一 下 会 
发 生 一 些 什么 事情 。 在 main0 中 会 创建 一 个 SerialCt 对 象 ， 随 后 会 序列 
化 到 一 个 ObjectOutputStream 里 〈 注 意 这 种 情况 下 使 用 的 是 一 个 缓冲 
区 ， 而 非 文 件 与 ObjectOutputStream 完 全 一 人 改 ) 。 正 式 的 序列 化 操 
作 是 在 下 面 这 行 代码 里 发 生 的 : 

















o.writeObject(sc); 


其 中 ，writeObject(0) 方 法 必须 核查 gc， 判断 它 是 否 有 目 己 的 writeObjectO) 
方法 (不 是 检查 它 的 接口 它 根 本 束 没 有 ， 也 不 是 检查 类 的 类 型 ， 而 
是 利用 反射 方法 实际 搜索 方法 ) 。 大 答案 是 肯定 的 ， 就 使 用 那个 方法 。 
类 似 的 情况 也 会 在 readObject0 上 人 有 发生。 或 许 这 是 解决 问题 唯一 实际 的 方 
法 ， 但 确实 显得 有 些 古 怪 。 











3. 版 本 问题 


有 时 候 可 能 想 改 变 一 个 可 序列 化 的 类 的 版 本 《比如 原始 类 的 对 象 可 能 保 
存在 数据 库 中 ) 。 尽 管 这 种 做 法 得 到 了 支持 ， 但 一 般 只 应 在 非常 特殊 的 
情况 下 才 用 它 。 此 外 ， 它 要 求 操作 者 对 背后 的 原理 有 一 个 比较 深 的 认 

识 ， 而 我 们 在 这 里 还 不 想 达 到 这 种 深度 。JDK 1.1 的 HTML 文 档 对 这 一 主 
题 进行 了 非常 全 面 的 论述 (可 从 Sun 公 司 下 载 ， 但 可 能 也 成 了 Java 开 发 

包 联 机 文档 的 一 部 分 ) 。 

10.9.3 利用 “持久 性 ” 

一 个 比较 诱 人 的 想法 是 用 序列 化 技术 保存 程序 的 一 些 状态 信息 ， 从 而 将 
程序 方便 地 恢复 到 以 前 的 状态 。 但 在 具体 实现 以 前 ， 有 些 问 题 是 必须 解 
决 的 。 如 果 两 个 对 象 都 有 指向 第 三 个 对 象 的 句柄 ， 该 如 何 对 这 两 个 对 象 
序列 化 呢 ? 如 果 从 两 个 对 象 序列 化 后 的 状态 恢复 它们 ， 第 三 个 对 象 的 句 
柄 只 会 出 现在 一 个 对 象 身 上 吗 ? 如 果 将 这 两 个 对 象 序列 化 成 独立 的 文 

件 ， 然 后 在 代码 的 不 同 部 分 重新 装配 它们 ， 又 会 得 到 什么 结果 呢 ? 


下 面 这 个 例子 对 上 述 问题 进行 了 很 好 的 说 明 : 


//: MyWorld.java 














import java.io.*; 
import java.util.*; 
class House implements Serializable {} 
class Animal implements Serializable { 
String name; 
House preferredHouse; 
Animal(String nm, House h) { 
name = nm; 


preferredHouse = h; 


} 
public String toString() { 
return name + "[" + super.toString() + 


"], " + preferredHouse + "\n"; 


} 
public class MyWorld { 
public static void main(String[] args) { 
House house = new House(); 
Vector animals = new Vector(); 
animals .addElement ( 
new Animal("Bosco the dog", house)); 
animals.addElement ( 
new Animal("Ralph the hamster", house)); 
animals.addElement ( 
new Animal("Fronk the cat", house)); 
System.out.println("animals: " + animals); 
try { 
ByteArrayOutputStream buf1 = 
new ByteArrayOutputStream(); 
ObjectOutputStream o1 = 
new ObjectOutputStream(buf1); 


o1.writeObject(animals); 


o1.writeObject(animals); // Write a 2nd set 
// Write to a different stream: 
ByteArrayOutputStream buf2 = 
new ByteArrayOutputStream(); 
ObjectOutputStream o2 = 
new ObjectOutputStream(buf2); 
02.writeObject(animals); 
// Now get them back: 
ObjectInputStream ini = 
new ObjectInputStream( 
new ByteArrayInputStream( 
buf1.toByteArray())); 
ObjectInputStream in2 = 
new ObjectInputStream( 
new ByteArrayInputStream( 
buf2.toByteArray())); 
Vector animals1 = (Vector)in1i.readObject(); 
Vector animals2 = (Vector)in1i.readObject(); 


Vector animals3 = (Vector)in2.readObject(); 


System.out.println("animalsi1: " + animals1); 
System.out.printlin("animals2: " + animals2); 
System.out.printin("animals3: " + animals3); 


} catch(Exception e) { 


e.printStackTrace(); 


} 
A f/f 





这 里 一 件 有 趣 的 事情 是 也 许 是 能 针对 一 个 字 节 数组 应 用 对 象 的 序列 化 ， 
从 而 实现 对 任何 Serializable (可 序列 化 ) 对 象 的 一 个 “全 面 复制 ”( 全 面 
复制 意味 着 复制 的 是 整个 对 象 网 ， 而 不 仅 是 基本 对 象 和 和 它 的 句柄 ) 。 复 
制 问题 将 在 第 12 章 进行 全 面 讲述 。 

Animal 对 象 包含 了 类 型 为 House 的 字段 。 在 main0) 中 ， 会 创建 这 些 
Animal 的 一 个 Vector， 并 对 其 序列 化 两 次 ， 分 别 送 入 两 个 不 同 的 数据 流 
内 。 这 些 数据 重新 装配 并 打印 出 来 后 ， 可 看 到 下 面 这 样 的 结果 (对 象 在 
每 次 运行 时 都 会 处 在 不 同 的 内 存 位 置 ， 所 以 每 次 运行 的 结果 有 区 别 〉: 


animals: [Bosco the dog[Animal@icc76c], House@1cc769 

















, Ralph the hamster[Animal@icc76d], House@1cc769 

, Fronk the cat[Animal@icc76e], House@1icc769 

] 

animalsi: [Bosco the dog[Animal@1cca0c], House@iccai6 
, Ralph the hamster[Animal@1icca17], House@1cca16 

, Fronk the cat[Animal@iccaib], House@1icca16 

] 

animals2: [Bosco the dog[Animal@1cca0c], House@iccai6 
, Ralph the hamster[Animal@1icca17], House@1cca16 


, Fronk the cat[Animal@iccaib], House@icca16 


] 


animals3: [Bosco the dog[Animal@1cca52], House@1icca5c 
, Ralph the hamster [Animal@icca5d], House@1cca5c 


, Fronk the cat[Animal@icca61], House@icca5c 





当然 ， 我 们 希望 装配 好 的 对 象 有 与 原来 不 同 的 地 址 。 但 注意 在 animals1 
和 animals2 中 出 现 了 相同 的 地 址 ， 其 中 包括 共享 的 、 对 House 对 象 的 引 
用 。 在 另 一 方面 ， 当 animals3 恢 复 以 后 ， 系 统 没 有 办 法 知道 另 一 个 流 内 
的 对 象 是 第 一 个 流 内 对 象 的 化 号 ， 所 以 会 产生 一 个 完全 不 同 的 对 象 网 。 


只 要 将 所 有 东西 都 序列 化 到 单独 一 个 数据 流 里 ， 束 能 恢复 获得 与 以 前 写 
入 时 完全 一 样 的 对 象 网 ， 不 会 不 慎 造 成 对 象 的 重复 。 当 然 ， 在 写 第 一 个 
和 最 后 一 个 对 象 的 时 间 之 间 ， 可 改变 对 象 的 状态 ， 但 那 必须 由 我 们 明确 
采取 操作 一 一 序列 化 时 ， 对 象 会 采用 它们 当时 的 任何 状态 (包括 它们 与 
其 他 对 象 的 连接 关系 ) 写 入 。 


奉 想 保存 系统 状态 ， 最 安全 的 做 法 是 当 作 一 种 “微观 ”操作 序列 化 。 如 果 
序列 化 了 某 些 东 西 ， 再 去 做 其 他 一 些 工 作 ， 再 来 序列 化 更 多 的 东西 ， 以 
此 类 推 ， 那 么 最 终 将 无 法 安全 地 保存 系统 状态 。 相 反 ， 应 将 构成 系统 状 
态 的 所 有 对 象 部 置 入 单个 集合 内 ， 并 在 一 次 操作 里 完成 那个 集合 的 写 
入 。 这 样 一 来 ， 同 样 只 需 一 次 方法 调用 ， 即 可 成 功 恢复 之 。 


下 面 这 个 例子 是 一 套 假想 的 计算 机 辅助 设计 CCAD) 系统 ， 对 这 一 方法 
进行 了 很 好 的 演示 。 此 外 ， 它 还 为 我 们 引入 了 static 字 段 的 问题 一 一 如 留 
意 联机 文档 ， 就 会 发 现 Class 是 “Serializable”(〈 可 序列 化 ) 的 ， 所 以 只 需 
简单 地 序列 化 Class 对 象 ， 就 能 实现 static 字 段 的 保存 。 这 无 论 如 何 都 是 
一 种 明智 的 做 法 。 


//: CADState. java 

















// Saving and restoring the state of a 
// pretend CAD system. 


import java.io.*; 


import java.util.*; 
abstract class Shape implements Serializable { 
public static final int 
RED = 1, BLUE = 2, GREEN = 3; 
private int xPos, yPos, dimension; 
private static Random r = new Random(); 
private static int counter = 0; 
abstract public void setColor(int newColor); 
abstract public int getColor(); 
public Shape(int xVal, int yVal, int dim) { 
xPos = xVal; 
yPos = yVal; 
dimension = dim; 
} 
public String toString() { 
return getClass().toString() + 
" color[" + getColor() + 
"] xPos[" + xPos + 
"] yPos[" + yPos + 
"] dim[" + dimension + "J\n"; 
} 
public static Shape randomFactory() { 


int xVal = r.nextInt() % 100; 


int yVal = r.nextInt() % 100; 

int dim = r.nextInt() % 100; 

Switch(counter++ % 3) { 
default: 
case 0: return new Circle(xVal, yVal, dim); 
case 1: return new Square(xVal, yVal, dim); 


case 2: return new Line(xVal, yVal, dim); 


} 
class Circle extends Shape { 
private static int color = RED; 
public Circle(int xVal, int yVal, int dim) { 
super(xVal, yVal, dim); 
} 
public void setColor(int newColor) { 
color = newColor; 
} 
public int getColor() { 


return color; 


} 


class Square extends Shape { 


private static int color; 

public Square(int xVal, int yVal, int dim) { 
super(xVal, yVal, dim); 
color = RED; 

} 

public void setColor(int newColor) { 
color = newColor; 

} 

public int getColor() { 


return color; 


class Line extends Shape { 

private static int color = RED; 

public static void 

serializeStaticState(ObjectOutputStream os) 
throws IOException { 

os.writeInt(color); 

} 

public static void 

deserializeStaticState(ObjectInputStream os) 
throws IOException { 


color = os.readInt(); 


} 

public Line(int xVal, int yVal, int dim) 
super(xVal, yVal, dim); 

} 

public void setColor(int newColor) { 
color = newColor; 

} 

public int getColor() { 


return color; 


J 


public class CADState { 
public static void main(String[] args) 
throws Exception { 
Vector shapeTypes, shapes; 
if(args.length == 0) { 
shapeTypes = new Vector(); 
shapes = new Vector(); 
// Add handles to the class objects: 
shapeTypes.addElement(Circle.class); 
shapeTypes.addElement(Square.class); 
shapeTypes.addElement(Line.class); 


// Make some shapes: 


for(int i = 0; i < 10; i++) 


shapes.addElement(Shape.randomFactory()); 


// Set all the static colors to GREEN: 


for(int i = 0; i < 10; i++) 


((Shape)shapes.elementAt(i) ) 
.setColor(Shape.GREEN) ; 
// Save the state vector: 
ObjectOutputStream out = 
new ObjectOutputStream( 
new FileOutputStream("CADState.out")); 
out.writeObject(shapeTypes); 
Line.serializeStaticState(out); 
out.writeObject(shapes); 
} else { // There's a command-line argument 
ObjectInputStream in = 
new ObjectInputStream( 
new FileInputStream(args[0])); 
// Read in the same order they were written: 
shapeTypes = (Vector)in.readObject(); 
Line.deserializeStaticState(in); 
shapes = (Vector)in.readObject(); 


} 


// Display the shapes: 


System.out.println(shapes) ) 


} 
i/o 


Shape (几何 形状 〉 类 “实现 了 可 序列 化 ”(implements Serializable) ， 所 
以 从 Shape 继 承 的 任何 东西 也 都 会 自动 “可 序列 化 ”。 每 个 Shape 都 包含 了 
数据 ， 而 且 每 个 久生 的 Shape 类 都 包含 了 一 个 特殊 的 static 字 段 ， 用 于 决 
定 所 有 那些 类 型 的 Shape 的 颜色 (如 将 一 个 static 字 段 置 入 基础 类 ， 结 果 
只 会 产生 一 个 字段 ， 因 为 static 字 段 未 在 衍生 类 中 复制 ) 。 可 对 基础 类 中 
的 方法 进行 覆盖 处 理 ， 以 便 为 不 同 的 类 型 设置 颜色 〈static 方 法 不 会 动态 
绑 定 ， 所 以 这 些 都 是 普通 的 方法 ) 。 每 次 调用 randomFactory() 方 法 时 ， 
它 都 会 创建 一 个 不 同 的 Shape (Shape KH IENE) 。 


Circle〈 圆 ) 和 Square 〈 和 矩形 ) 属于 对 Shape 的 直接 扩展 ; 唯一 的 差别 是 
Circle 在 定义 时 会 初始 化 颜色 ， 而 Square 在 构建 器 中 初始 化 。Line CA 
线 ) 的 问题 将 留 到 以 后 讨论 。 


在 main0 中 ， 一 个 Vector 用 于 容纳 Class 对 象 ， 而 另 一 个 用 于 容纳 形状 。 
苛 不 提供 相应 的 命令 行 参 数 ， 就 会 创建 shapeTypes Vector， 并 添加 Class 
对 象 。 然 后 创建 shapes Vector， 并 添加 Shape 对 象 。 接 下 来 ， 所 有 static 
color 值 都 会 设 成 GREEN， 而 且 所 有 东西 都 会 序列 化 到 文件 
CADState.out. 


若 提 供 了 一 个 命令 行 参数 (假设 CADState.out) ， 便 会 打开 那个 文件 ， 
并 用 它 恢复 程序 的 状态 。 无 论 在 哪 种 情况 下 ， 结 果 产 生 的 Shape 的 Vector 
都 会 打印 出 来 。 下 面 列 出 它 茶 一 次 运行 的 结 


>java CADState 


[class Circle color[3] xPos[-51] yPos[-99] dim[38] 
, Class Square color[3] xPos[2] yPos[61] dim[ -46] 
, Class Line color[3] xPos[51] yPos[73] dim[64] 


, Class Circle color[3] xPos[-70] yPos[1] dim[16] 


, Class Square color[3] xPos[3] yPos[94] dim[-36] 

, Class Line color[3] xPos[-84] yPos[-21] dim[-35] 
, Class Circle color[3] xPos[-75] yPos[-43] dim[22] 
, Class Square color[3] xPos[81] yPos[30] dim[-45] 
, Class Line color[3] xPos[-29] yPos[92] dim[17] 

, Class Circle color[3] xPos[17] yPos[90] dim[-76] 
] 

>java CADState CADState.out 

[class Circle color[1] xPos[-51] yPos[-99] dim[38] 
, Class Square color[0] xPos[2] yPos[61] dim[ -46] 

, Class Line color[3] xPos[51] yPos[73] dim[64] 

, Class Circle color[1] xPos[-70] yPos[1] dim[16] 

, Class Square color[0] xPos[3] yPos[94] dim[ -36] 

, Class Line color[3] xPos[-84] yPos[-21] dim[-35] 
, Class Circle color[1] xPos[-75] yPos[-43] dim[22] 
, Class Square color[0] xPos[81] yPos[30] dim[-45] 
, Class Line color[3] xPos[-29] yPos[92] dim[17] 


, Class Circle color[1] xPos[17] yPos[90] dim[-76] 





从 中 可 以 看 出 ，xPos，yPos 以 及 dim 的 值 都 已 成 功 保存 和 恢复 出 来 。 但 
在 获取 static 信 息 时 却 出 现 了 问题 。 所 有 “3” 都 已 进入 ， 但 没有 正常 地 出 
来 。Circle 有 一 个 1 值 (定义 为 RED) ， 而 Square 有 一 个 0 值 〈 记 住 ， 它 
们 是 在 构建 器 里 初始 化 的 ) 。 看 上 去 似乎 static 根 本 没有 得 到 初始 化 ! 实 
情 正 是 如 此 一 一 尽管 类 Class 是 “可 以 序列 化 的 ”， 但 却 不 能 按 我 们 希望 的 





工作 。 所 以 假如 想 序 列 化 static 值 ， 必 须 杀 目 动 手 。 


这 正 是 Line 中 的 serializeStaticState() 和 deserializeStaticState() 两 个 static 方 
法 的 用 途 。 可 以 看 到 ， 这 两 个 方法 都 是 作为 存储 和 恢复 进程 的 一 部 分 明 
确 调 用 的 (注意 写 入 序列 化 文件 和 从 中 读 回 的 顺序 不 能 改变 ) 。 所 以 为 
了 使 CADState.java 正 确 运行 起 来 ， 必 须 采 用 下 述 三 种 方法 之 一 : 








(1) 为 几何 形状 添加 一 个 serializeStaticState0 和 deserializeStaticState()。 

(2) 删除 Vector shapeTypes 以 及 与 之 有 关 的 所 有 代码 

(3) 在 几何 形状 内 添加 对 新 序列 化 和 撤消 序列 化 静态 方法 的 调用 

要 注意 的 另 一 个 问题 是 安全 ， 因 为 序列 化 处 理 也 会 将 private 数 据 保 存 下 
来 。 奋 有 需要 保密 的 字段 ， 应 将 其 标记 成 transient。 但 在 这 之 后 ， 必 须 
设计 一 种 安全 的 信息 保存 方法 。 这 样 一 来 ， 一 旦 需要 恢复 ， 就 可 以 重 设 


那些 private 变 量 。 











10.10 总 结 


Java IO 流 库 能 满足 我 们 的 许多 基本 要 求 : 可 以 通过 控制 台 、 文 件 、 内 存 
块 甚至 因特网 〈 参 见 第 15 章 ) 进行 读 写 。 可 以 创建 新 的 输入 和 输出 对 象 
类 型 (通过 从 InputStream 和 OutputStream 继 承 )。 问 一 个 本 来 预期 为 收 
到 字 串 的 方法 传递 一 个 对 象 时 ， 由 于 Java 已 限制 了 * 目 动 类 型 转换 ?”， 上 所 
以 会 自动 调用 toString() 方 法 。 而 我 们 可 以 重新 定义 这 个 toString0， 扩 展 
一 个 数据 流 能 接纳 的 对 象 种 类 。 


在 IO 数 据 流 库 的 联机 文档 和 设计 过 程 中 ， 仍 有 些 问题 没有 解决 。 比 如 当 
我 们 打开 一 个 文件 以 便 输出 时 ， 完 全 可 以 指定 一 旦 有 人 试图 宪 盖 该 文件 
就 “ 撕 ” 出 一 个 违例 一 一 有 的 编程 系统 允许 我 们 自行 指定 想 打开 一 个 输出 
文件 ， 但 唯一 的 前 提 是 它 尚 不 存在 。 但 在 Java 中 ， 似 乎 必须 用 一 个 File 
对 象 来 判断 某 个 文件 是 否 存在 ， 因 为 假如 将 其 作为 FileOutputStream 或 者 
FileWriter 打 开 ， 那 么 肯定 会 被 履 盖 。 若 同时 指定 文件 和 目录 路 径 ，File 
类 设计 上 的 一 个 缺陷 承 会 骏 露 出 来 ， 因 为 它 会 说 “不 要 试图 在 单个 类 里 
做 太 多 的 事情 ”! 


IO 流 库 易 使 我 们 混淆 一 些 概念 。 它 确实 能 做 许多 事情 ， 而 且 也 可 以 移 

植 。 但 假如 假如 事先 没有 吃透 装饰 器 方案 的 概念 ， 那 么 所 有 的 设计 都 多 
少 带 有 一 点 盲目 性 质 。 所 以 不 管 学 它 还 是 教 它 ， 都 要 特别 花 一 些 功 夫 才 
行 。 而 且 它 并 不 完整 : 没有 提供 对 输出 格式 化 的 文 持 ， 而 其 他 几乎 所 有 
语言 的 10 包 都 提供 了 这 方面 的 支持 (这 一 点 没有 在 Java 1.1 里 得 以 纠 
正 ， 它 完全 错失 了 改变 库 设计 方案 的 机 会 ， 反 而 增添 了 更 特殊 的 一 些 情 
况 ， 使 复杂 程度 进一步 提高 ) 。Java 1.1 转 到 那些 尚未 替换 的 IO 库 ， 而 
不 是 增加 新 库 。 而 且 库 的 设计 人 员 似 乎 没有 很 好 地 指出 哪些 特性 是 不 赞 
成 的 ， 哪 些 是 首选 的 ， 造 成 库 设 计 中 经 常 都 会 出 现 一 些 令 人 恼火 的 反对 


YL 
消息 。 


然而 ， 一 旦 掌握 了 装饰 希 方 案 ， 并 开始 在 一 些 较为 灵活 的 环境 使 用 库 ， 
就 会 认识 到 这 种 设计 的 好 处 。 到 那个 时 候 ， 为 此 多 付出 的 代码 行 应 该 不 
至 于 使 你 觉得 太 生 气 。 




















10.11 练习 


(1) ”打开 一 个 文本 文件 ， 每 次 恋 取 一 行内 容 。 将 每 行 作为 一 个 String 读 
入 ， 并 将 那个 String 对 象 置 入 一 个 Vector 里 。 按 相反 的 顺序 打印 出 Vector 
中 的 所 有 行 。 


(2) 修改 练习 1， 使 读 取 那 个 文件 的 名 字 作 为 一 个 命令 行 参数 提供 。 

(3) 修改 练习 2， 又 打开 一 个 文本 文件 ， 以 便 将 文字 写 入 其 中 。 将 Vector 
中 的 行 随 同行 号 一 起 写 入 文件 。 

(4) ”修改 练习 2， 强 迫 Vector 中 的 所 有 行 都 变 成 大 写 形 式 ， 将 结果 友 给 
System.out. 

(5) 修改 练习 2， 在 文件 中 碍 找 指定 的 单词 。 打 印 出 包含 了 欲 找 单 词 的 所 
有 文本 行 。 


(6) ”在 Blips.java 中 复制 文件 ， 将 其 重 命 名 为 BlipCheck.java。 然 后 将 类 
Blip2 重 命名 为 BlipCheck《〈 在 进程 中 将 其 标记 为 public) 。 删 除 文件 中 
的 /记号 ， 并 执行 程序 。 接 下 来 ， 将 BlipCheck 的 默认 构建 器 变 成 注释 信 
息 。 运 行 它 ， 并 解释 为 什么 仍然 能 够 工作 。 


(7) 在 Blip3.java 中 ， 将 接 在 “You must do this:” 字 样 后 的 两 行 变 成 注释 ， 
然后 运行 程序 。 解 释 得 到 的 结果 为 什么 会 与 执行 了 那 两 行 代码 不 同 。 
(8) 转换 SortedWordCount.java 程 序 ， 以 便 使 用 Java 1.1 IO 流 。 

(9) 根据 本 章 正 文 的 说 明 修 改 程序 CADState.java。 


(10) 在 第 7 章 〈 中 间 部 分 ) 找到 GreenhouseControls.java 示 例 ， 它 应 该 由 
三 个 文件 构成 。 在 GreenhouseControls.java 中 ，Restart() 内 部 类 有 一 个 便 
编码 的 事件 集 。 请 修改 这 个 程序 ， 使 其 能 从 一 个 文本 文件 里 动态 读 取 事 
件 以 及 它们 的 相关 时 间 。 




















FALE 运行 期 类 型 鉴定 


运行 期 类 型 鉴定 (RTTI) 的 概念 初 看 非常 简单 一 一 手 上 只 有 基础 类 型 
的 一 个 句柄 时 ， 利 用 它 判 断 一 个 对 象 的 正确 类 型 。 


然而 ， 对 RTTI 的 需要 暴露 出 了 面 癌 对 象 设计 许多 有 趣 CM AA ES 
人 困惑 的 ) 的 问题 ， 并 把 程序 的 构造 问题 正式 摆 上 了 蝎 面 。 


本 章 将 讨论 如 何 利用 Java 在 运行 期 间 奉 找 对 象 和 类 信息 。 这 主要 采取 两 
种 形式 : 一 种 是 “传统 "RTTI， 它 假定 我 们 已 在 编译 和 运行 期 拥有 所 有 类 
型 ， 男 一 种 是 Javal.1 特 有 的 “有 反射” 机制， 利用 它 可 在 运行 期 独立 查找 类 
信息 。 首 先 讨论 “ 传 统 ”" 的 RTTI， 再 讨论 反射 问题 。 














11.1 对 RTTI 的 需要 


请 考虑 下 面 这 个 熟悉 的 类 结构 例子 ， 它 利用 了 多 形 性 。 和 常规 类 型 是 
Shape 类 ， 而 特别 衍生 出 来 的 类 型 是 Circle，Square 和 Triangle。 





这 是 一 个 典型 的 类 结构 示意 图 ， 基 础 类 位 于 顶部 ， 衍 生 类 向 下 延展 。 面 
回 对 象 编 程 的 基本 目标 是 用 大 量 代 码 控制 基础 类 型 〈 这 里 是 Shape) 的 
句柄 ， 所 以 假如 决定 添加 一 个 新 类 《比如 Rhomboid， 从 Shape 衍 生 ) , 
从 而 对 程序 进行 扩展 ， 那 么 不 会 影响 到 原来 的 代码 。 在 这 个 例子 中 ， 
Shape 接 口中 的 动态 绑 定 方法 是 draw0， 上 所 以 客户 程序 员 要 做 的 是 通过 一 
个 普通 Shape 人 句柄 调用 draw0。draw0 在 所 有 衍生 类 里 都 会 被 履 盖 。 而 且 
由 于 它 是 一 个 动态 绑 定 方法 ， 所 以 即使 通过 一 个 普通 的 Shape 人 句柄 调用 
它 ， 也 有 表现 出 正确 的 行为 。 这 正 是 多 形 性 的 作用 。 

所 以 ， 我 们 一 般 创 建 一 个 特定 的 对 象 〈Circle，Square， 或 者 

Triangle) ， 把 它 上 渊 造型 到 一 个 Shape〈 忽 略 对 象 的 特殊 类 型 ) ， 以 后 
便 在 程序 的 剩余 部 分 使 用 匿名 Shape 和 句柄 。 


作为 对 多 形 性 和 上 调 造 型 的 一 个 简要 回顾 ， 可 以 象 下 面 这 样 为 上 述 例子 
编码 〈 奋 执行 这 个 程序 时 出 现 困难 ， 请 参考 第 3 章 3.1.2 小 节 “ 赋 值 >) : 


//: Shapes.java 
































package c11; 
import java.util.*; 
interface Shape { 


void draw(); 


} 
class Circle implements Shape { 
public void draw() { 


System.out.println("Circle.draw()"); 


} 
class Square implements Shape { 
public void draw() { 


System.out.println("Square.draw()"); 


J 


class Triangle implements Shape { 
public void draw() { 


System.out.println("Triangle.draw()"); 


} 
public class Shapes { 
public static void main(String[] args) { 
Vector s = new Vector(); 
s.addElement(new Circle()); 
s.addElement(new Square()); 
s.addElement(new Triangle()); 


Enumeration e = s.elements(); 


while(e.hasMoreElements() ) 


((Shape)e.nextElement()).draw(); 


} 
A f/f 


基础 类 可 编码 成 一 个 interface (#201) 、 一 个 abstract (FHA) 类 或 者 一 
个 普通 类 。 由 于 Shape 没 有 真正 的 成 员 《〈 亦 即 有 定义 的 成 员 ) ， 而 且 并 

不 在 意 我 们 创建 了 一 个 纯粹 的 Shape 对 象 ， 所 以 最 适合 和 最 灵活 的 表达 

方式 便 是 用 一 个 接口 。 而 且 由 于 不 必 设 置 所 有 那些 abstract 关 键 字 ， 上 所 以 
整个 代码 也 显得 更 为 清 碍 。 


每 个 衍生 类 都 履 盖 了 基础 类 draw 方 法 ， 所 以 具有 不 同 的 行为 。 在 main0) 
中 创建 了 特定 类 型 的 Shape， 然 后 将 其 添加 到 一 个 Vector。 这 里 正 是 上 漳 
造型 发 生 的 地 方 ， 因 为 Vector 只 容纳 了 对 象 。 由 于 Java 中 的 所 有 东西 

( 除 基 本 数据 类 型 外 ) 都 是 对 象 ， 所 以 Vector 也 能 容纳 Shape 对 象 。 但 在 
上 泣 造 型 至 Object 的 过 程 中 ， 任 何 特殊 的 信息 都 会 丢失 ， 其 中 甚至 包括 
对 象 是 几何 形状 这 一 事实 。 对 Vector 来 说 ， 它 们 只 是 Object。 


用 nextElementO 将 一 个 元 素 从 Vector 提取 出 来 的 时 候 ， 情 况 变 得 稍微 有 
些 复杂 。 由 于 Vector 只 容纳 Object， 所 以 nextElement0 会 自然 地 产生 一 个 
Object 句柄 。 但 我 们 知道 它 实 际 是 个 Shape 句 柄 ， 而 且 和 希望 将 Shape 消 息 
发 给 那个 对 象 。 所 以 需要 用 传统 的 "(Shape)" 方 式 造型 成 一 个 Shape。 这 
是 RTTI 最 基本 的 形式 ， 因 为 在 Java 中 ， 所 有 造型 都 会 在 运行 期 间 得 到 检 
E> UR AE BE 那 正 是 RTTI 的 意义 所 在 : 在 运行 期 ， 对 象 的 类 
会 得 到 鉴定 。 


在 目前 这 种 情况 下 ，RTTI 造 型 只 实现 了 一 部 分 : Object 造型 成 Shape， 
而 不 是 造型 成 Circle，Square 或 者 Triangle。 那 是 由 于 我 们 目前 能 够 肯定 
的 唯一 事实 就 是 Vector 里 充斥 着 几何 形状 ， 而 不 知 它们 的 有 具体 类 别 。 在 
编译 期 间 ， 我 们 肯定 的 依据 是 我 们 自己 的 规则 ;而 在 编译 期 间 ， 却 是 通 
过 造型 来 肯定 这 一 点 。 


现在 的 局 面 会 由 多 形 性 控制 ， 而 且 会 为 Shape 调 用 适当 的 方法 ， 以 便 判 
断 句 柄 到 底 是 提供 Circle，Square， 还 是 提供 给 Triangle。 而 且 在 一 般 情 
况 下 ， 必 须 保证 采用 多 形 性 方案 。 因 为 我 们 希望 目 己 的 代码 尽 可 能 少 知 






































道 一 些 与 对 象 的 具体 类 型 有 关 的 情况 ， 只 将 注音 力 放 在 某 一 类 对 象 ( 这 
里 是 Shape〉 的 常规 信息 上 。 只 有 这 样 ， 我 们 的 代码 才 更 易 实现 、 理 解 
以 及 修改 。 所 以 说 多 形 性 是 面向 对 象 程序 设计 的 一 个 常规 目标 。 


然而 ， 帮 碰 到 一 个 特殊 的 程序 设计 间 题 ， 只 有 在 知道 第 规 句柄 的 确切 类 
型 后 ， 才 能 最 容易 地 解决 这 个 问题 ， 这 个 时 候 叉 该 上 怎么 办 呢 ? 举 个 例子 
来 说 ， 我 们 有 时 候 想 让 自己 的 用 户 将 东 一 具体 类 型 的 几何 形状 (如 三 角 
É) 全 都 变 成 紫色 ， 以 便 突出 显示 它们 ， 并 快速 找 出 这 一 类 型 的 所 有 形 
状 。 此 时 便 要 用 到 RTTI 技 术 ， 用 它 查 询 某 个 Shape 句 柄 引用 的 准确 类 型 


KEATS o 











11.1.1 Class% 


为 理解 RTTI 在 Java 里 如 何 工 作 ， 首 先 必须 了 解 类 型 信息 在 运行 期 是 如 何 
表示 的 。 这 时 要 用 到 一 个 名 为 “Class 对 象 ” 的 特殊 形式 的 对 象 ， 其 中 包含 
了 与 类 有 关 的 信息 (有 时 也 把 它 叫 作 “ 元 类 ”) 。 事 实 上 ， 我 们 要 用 Class 
对 象 创 建 属 于 茶 个 类 的 全 部 “常规 ?或 “普通 ”对象 。 


对 于 作为 程序 一 部 分 的 每 个 类 ， 它 们 都 有 一 个 Class 对 象 。 换 言 之 ， 每 次 
写 一 个 新 类 时 ， 同 时 也 会 创建 一 个 Class 对 象 〈 更 恰当 地 说 ， 是 保存 在 一 
个 完全 同名 的 .class 文 件 中 ) 。 在 运行 期 ， 一 旦 我 们 想 生 成 那个 类 的 一 个 
对 象 ， 用 于 执行 程序 的 Java 虚 拟 机 GVM) 首先 就 会 检查 那个 类 型 的 
Class 对 象 是 否 已 经 载 入 。 奉 尚未 载 入 ，JVM 束 会 查找 同名 的 .class 文 

件 ， 并 将 其 载 入 。 所 以 Java 程 序 启动 时 并 不 是 完全 载 入 的 ， 这 一 点 与 许 
多 传统 语言 都 不 同 。 


一 旦 那个 类 型 的 Class 对 象 进 入 内 存 ， 束 用 它 创 建 那 一 类 型 的 所 有 对 象 。 


奉 这 种 说 法 多 少 让 你 产生 了 一 点 儿 迷 惑 ， 或 者 并 没有 真正 理解 它 ， 下 面 
这 个 示范 程序 或 许 能 提供 进一步 的 帮助 : 


//: SweetShop.java 




















// Examination of the way the class loader works 
class Candy { 


static { 


System.out.println("Loading Candy"); 


J 


class Gum { 
static { 


System.out.println("Loading Gum"); 


} 


class Cookie { 
static { 


System.out.println("Loading Cookie"); 


} 


public class SweetShop { 
public static void main(String[] args) { 
System.out.printin("inside main"); 
new Candy(); 
System.out.printin("After creating Candy"); 
try { 
Class.forName("Gum"); 
} catch(ClassNotFoundException e) { 


e.printStackTrace(); 


System.out .printlLn( 
"After Class.forName(\"Gum\")"); 
new Cookie(); 
System.out.printin("After creating Cookie"); 
} 
} ///:~ 





对 每 个 类 来 说 〈Candy，Gum 和 Cookie) ， 它 们 都 有 一 个 static 从 句 ， 用 
于 在 类 首次 载 入 时 执行 。 相 应 的 信息 会 打印 出 来 ， 告 诉 我 们 载 入 是 什么 
时 候 进 行 的 。 在 main0 中 ， 对 象 的 创建 代码 位 于 打印 语句 之 间 ， 以 便 侦 
测 载 入 时 间 。 


特别 有 趣 的 一 行 是 : 
Class.forName("Gum"); 


该 方法 是 Class《〈 即 全 部 Class 所 从 属 的 ) 的 一 个 static 成 员 。 而 Class 对 象 
和 其 他 任何 对 象 都 是 类 似 的 ， 所 以 能 够 获取 和 控制 它 的 一 个 句柄 CR 
模块 就 是 干 这 件 事 的 ) 。 为 获得 Class 的 一 个 句柄 ， 一 个 办 法 是 使 用 
forName()。 它 的 作用 是 取得 包含 了 目标 类 文本 名 字 的 一 个 String (注意 
拼写 和 大 小 写 ) 。 最 后 返回 的 是 一 个 Class 句 柄 。 


该 程序 在 某 个 JVM 中 的 输出 如 下 : 


inside main 








Loading Candy 

After creating Candy 
Loading Gum 

After Class.forName( "Gum" ) 


Loading Cookie 


After creating Cookie 


可 以 看 到 ， 每 个 Class 只 有 在 它 需 要 的 时 候 才 会 载 入 ， 而 static 初 始 化 工 
作 是 在 类 载 入 时 执行 的 。 
非常 有 趣 的 是 ， 另 一 个 JVM 的 输出 变 成 了 兄 一 个 样子 : 


Loading Candy 


Loading Cookie 

inside main 

After creating Candy 
Loading Gum 

After Class. forName("Gum" ) 


After creating Cookie 


看 来 JVM 通 过 检查 main0) 中 的 代码 ， 已 经 预测 到 了 对 Candy 和 Cookie 的 需 
要 ， 但 却 看 不 到 Gum， 因 为 它 是 通过 对 forName0 的 一 个 调用 创建 的 ， 

而 不 是 通过 更 典型 的 new 调 用 。 尽 管 这 个 JVM 也 达到 了 我 们 希望 的 效 
果 ， 因 为 确实 会 在 我 们 需要 之 前 载 入 那些 类 ， 但 却 不 能 肯定 这 儿 展 示 的 
行为 百分之百 正确 。 


1. 类 标记 


在 Java 1.1 中 ， 可 以 采用 第 二 种 方式 来 产生 Class 对 象 的 句柄 : 使 用 “类 标 
记 ”。 对 上 述 程序 来 说 ， 看 起 来 就 象 下 面 这 样 : 








Gum.class; 


这 样 做 不 仅 更 加 人 简单 ， 而 且 更 安全 ， 因 为 它 会 在 编 详 期 间 得 到 检查 。 由 
于 它 取 消 了 对 方法 调用 的 需要 ， 所 以 执行 的 效率 也 会 更 高 。 


类 标记 不 仅 可 以 应 用 于 普通 类 ， 也 可 以 应 用 于 接口 、 数 组 以 及 基本 数据 











类 型 。 除 此 以 外 ， 针 对 每 种 基本 数据 类 型 的 封装 器 类 ， 它 还 存在 一 个 名 
为 TYPE 的 标准 字段 。TYPE 字 上 段 的 作用 是 为 相关 的 基本 数据 类 型 产生 
Class 对 象 的 一 个 句柄 ， 如 下 所 示 : 


. IS equivalent to ... 


boolean.class 





Boolean. TYPE 
char.class 
Character. TYPE 
byte.class 
Byte. TYPE 
short.class 
Short. TYPE 
int.class 
Integer. TYPE 
long.class 
Long. TYPE 
float.class 
Float.TYPE 
double.class 
Double. TYPE 


void.class 


Void. TYPE 
11.1.2 造型 前 的 检查 
迄今 为 止 ， 我 们 已 知 的 RTTI 形 式 包括 ; 


(经 典 造型 ， 如 "(Shape)"， 它 用 RTTI 确 保 造型 的 正确 性 ， 并 在 遇 到 一 
个 失败 的 造型 后 产生 一 个 ClassCastException 违 例 。 


代表 对 象 类 型 的 Class 对 象 。 可 查询 Class 对 象 ， 获 取 有 用 的 运行 期 资 


在 C++ 中 ， 经 典 的 "(Shape)" 造 型 并 不 执行 RTTI。 它 只 是 简单 地 告诉 编译 
器 将 对 象 当 作 新 类 型 处 理 。 而 Java 要 执行 类 型 检查 ， 这 通常 叫 作 “ 类 型 

安全 ”的 下 渊 造型 。 之 所 以 叫 “ 下 渊 造型 >”， 是 由 于 类 分 层 结构 的 历史 排 

列 方式 造成 的 。 知 将 一 个 Circle (AD 造型 到 一 个 Shape( 几 何 形状 )， 

就 叫做 上 浏 造 型 ， 因 为 加 只 是 几何 形状 的 一 个 子 集 。 反 之 ， 寿 将 Shape 
造型 至 Circle， 束 叫做 下 济 造 型 。 然 而 ， 尺 管 我 们 明确 知道 Circle 也 是 一 
个 Shape， 所 以 编译 器 能 够 自动 上 济 造 型 ， 但 却 不 能 保证 一 个 Shape 肯 定 
是 一 个 Circle。 因 此 ， 编 译 堪 不 允许 自动 下 漳 造 型 ， 除 非 明确 指定 一 次 
这 样 的 造型 。 


RTTI 在 Java 中 存在 三 种 形式 。 关 键 字 instanceof 告 诉 我 们 对 象 是 不 是 一 个 
特定 类 型 的 实例 〈Instance 即 “实例 ”) 。 它 会 返回 一 个 布尔 值 ， 以 便 以 
问题 的 形式 使 用 ， 融 象 下 面 这 样 : 








if(x instanceof Dog) 

((Dog)x).barkQ); 

将 x 造型 至 一 个 Dog 前 ， 上 面 的 站 语句 会 检查 对 象 x 是 否 从 属于 Dog 类 。 进 
行 造 型 前 ， 如 果 没 有 其 他 信息 可 以 告诉 自己 对 象 的 类 型 ， 那 么 instanceof 
的 使 用 是 非常 重要 的 否则 会 得 到 一 个 ClassCastException 违 例 。 


我 们 最 一 般 的 做 法 是 查找 一 种 类 型 〈 比 如 要 变 成 紫色 的 三 角形 ) ， 但 下 
面 这 个 程序 却 演示 了 如 何 用 instanceof 标 记 出 所 有 对 象 。 


//: PetCount.java 





// Using instanceof 
package cii.petcount; 
import java.util.*; 
class Pet {} 
class Dog extends Pet {} 
class Pug extends Dog {} 
class Cat extends Pet {} 
class Rodent extends Pet {} 
class Gerbil extends Rodent {} 
class Hamster extends Rodent {} 
class Counter { int i; } 
public class PetCount { 
static String[] typenames = { 
"Pet", "Dog", "Pug", "Cat", 
"Rodent", "Gerbil", "Hamster", 
J; 
public static void main(String[] args) { 
Vector pets = new Vector(); 
try { 
Class[] petTypes = { 
Class.forName("c1i1.petcount.Dog"), 
Class.forName("c1i1.petcount.Pug"), 


Class.forName("ci1.petcount.Cat"), 


Class. forName("c1i1.petcount.Rodent"), 
Class. forName("c1i1.petcount.Gerbil"), 
Class. forName("c1i1.petcount.Hamster"), 
}; 
for(int i = 0; i < 15; i++) 
pets.addElement ( 
petTypes[ 
(int )(Math.random()*petTypes. length) ] 
.newInstance()); 

} catch(InstantiationException e) {} 
catch(IllegalAccessException e) {} 
catch(ClassNotFoundException e) {} 

Hashtable h = new Hashtable(); 

for(int i = 0; i < typenames.length; i++) 
h.put(typenames[i], new Counter()); 

for(int i = 0; i < pets.size(); i++) { 
Object o = pets.elementAt(i); 
if(o instanceof Pet) 

((Counter )h.get("Pet")).1i++; 
if(o instanceof Dog) 

((Counter )h.get("Dog")).1i++; 
if(o instanceof Pug) 


((Counter )h.get("Pug")).i++; 


if(o instanceof Cat) 
((Counter )h.get("Cat")).i++; 
if(o instanceof Rodent) 
((Counter )h.get("Rodent")).it++; 
if(o instanceof Gerbil) 
((Counter )h.get("Gerbil")).it++; 
if(o instanceof Hamster) 
((Counter )h.get("Hamster")).1++; 
} 
for(int i = 0; i < pets.size(); i++) 


System. out.printin( 


pets.elementAt(i).getClass().toString()); 


for(int i = 0; i < typenames.length; i++) 


System. out.printin( 


typenames[i] + " quantity: " + 


((Counter )h.get(typenames[i])).1i); 


} 
} ///:~ 


在 Java ”1.0 中 ， 对 instanceof 有 一 个 比较 小 的 限制 : 


只 可 将 其 与 一 个 已 命 


名 的 类 型 比较 ， 不 能 同 Class 对 象 作 对 比 。 在 上 述 例子 中 ， 大 家 可 能 觉得 








将 所 有 那些 instanceof 表 达 式 写 出 来 是 件 很 麻烦 的 事情 。 实 际 情况 正 是 这 





样 。 但 在 Java 1.0 中， 没有 办 法 让 这 一 工作 目 动 进行 


不 能 创建 Class 





的 一 个 Vector， 再 将 其 与 之 比较 。 大 家 最 终 会 意识 到 ， 如 编写 了 数量 众 


多 的 instanceof 表 达 式 ， 整 个 设计 都 可 能 出 现 问题 。 


当然 ， 这 个 例子 只 是 一 个 构想 一 一 最 好 在 每 个 类 型 里 添加 一 个 static 数 据 
成 员 ， 然 后 在 构建 器 中 令 其 增值 ， 以 便 跟 踪 计 数 。 编 写 程序 时 ， 大 家 可 

能 想象 自己 拥有 类 的 源码 控制 权 ， 能 够 自由 改动 它 。 但 由 于 实际 情况 并 
非 总 是 这 样 ， 所 以 RTTI 显 得 特别 方便 。 


1. 使 用 类 标记 


PetCount.java 示 例 可 用 Java 1.1 的 类 标记 重 写 一 过 。 得 到 的 结果 显得 更 加 
明确 易 懂 : 


//: PetCount2.java 





// Using Java 1.1 class literals 
package cii.petcount2; 
import java.util.*; 
class Pet {} 
class Dog extends Pet {} 
class Pug extends Dog {} 
class Cat extends Pet {} 
Class Rodent extends Pet {} 
class Gerbil extends Rodent {} 
class Hamster extends Rodent {} 
class Counter { int i; } 
public class PetCount2 { 
public static void main(String[] args) { 
Vector pets = new Vector(); 


Class[] petTypes = { 


// Class literals work in Java 1.1+ only: 
Pet.class, 
Dog.class, 
Pug.class, 
Cat.class, 
Rodent.class, 
Gerbil.class, 
Hamster.class, 
}; 
try { 
for(int i = 0; i < 15; i++) { 


// Offset by one to eliminate Pet.class: 


int rnd = 1 + (int)( 


Math.random() * (petTypes.length - 1)); 
pets.addElement ( 
petTypes[rnd].newInstance()); 
} 
} catch(InstantiationException e) {} 
catch(IllegalAccessException e) {} 
Hashtable h = new Hashtable(); 
for(int i = 0; i < petTypes.length; i++) 
h.put(petTypes[i].toString(), 


new Counter()); 


for(int i = 0; i < pets.size(); i++) { 
Object o = pets.elementAt(i); 
if(o instanceof Pet) 
((Counter )h. get ( 
"Class cii.petcount2.Pet")).i++; 
if(o instanceof Dog) 
((Counter )h. get ( 
"Class cii.petcount2.Dog")).i++; 
if(o instanceof Pug) 
((Counter )h. get ( 
"Class cii1.petcount2.Pug")).i++; 
if(o instanceof Cat) 
((Counter )h. get ( 
"Class ciil.petcount2.Cat")).i++; 
if(o instanceof Rodent) 
((Counter )h. get ( 
"Class cii1.petcount2.Rodent")).i++; 
if(o instanceof Gerbil) 
((Counter )h. get ( 
"Class cii1.petcount2.Gerbil")).i++; 
if(o instanceof Hamster) 
((Counter )h. get ( 


"Class cii1.petcount2.Hamster")).i++; 


} 
for(int i = 0; i < pets.size(); i++) 
System.out.printin( 
pets.elementAt(i).getClass().toString()); 
Enumeration keys = h.keys(); 
while(keys.hasMoreElements()) { 
String nm = (String)keys.nextElement(); 
Counter cnt = (Counter)h.get(nm); 
System.out.printin( 
nm.substring(nm.lastIndexOf('.') + 1) + 


" quantity: " + cnt.1); 


} 
} ///:~ 


在 这 里 ，typenames (类 型 名 ) 数组 已 被 删除 ， 改 为 从 Class 对 象 里 获取 
类 型 名 称 。 注 意 为 此 而 额外 做 的 工作 : 例如 ， 类 名 不 是 Getbil， 而 是 
cl1.petcount2.Getbil， 其 中 已 包含 了 包 的 名 字 。 也 要 注意 系统 是 能 够 区 
分 类 和 接口 的 。 


也 可 以 看 到 ，petTypes 的 创建 模块 不 需要 用 一 个 try 块 包围 起 来 ， 因 为 它 
会 在 编译 期 得 到 检查 ， 不 会 象 Class.forName0) 那 样 “ 搓 ?出 任何 违例 。 


Pet 动态 创建 好 以 后 ， 可 以 看 到 随机 数字 已 得 到 了 限制 ， 位 于 1 和 
petTypes.length 之 间 ， 而 且 不 包括 零 。 那 是 由 于 零 代 表 的 是 Pet.class， 而 
且 一 个 普通 的 Pet 对 象 可 能 不 会 有 人 感 兴 趣 。 然 而 ， 由 于 Pet.class 是 
petTypes 的 一 部 分 ， 所 以 所 有 Pet《〈 宠 物 ) 都 会 算 入 计数 中 。 














2. 动态 的 instanceof 





Java 1.1 为 Class 类 添加 了 isInstance 方 法 。 利 用 它 可 以 动态 调用 instanceof 
运算 符 。 而 在 Java 1.0 中 ， 只 能 静态 地 调用 它 〈 就 象 前 面 指 出 的 那 
RE) 。 因 此 ， 所 有 那些 烦人 的 instanceof 语 句 都 可 以 从 PetCount 例 子 中 删 
去 了 。 如 下 所 示 : 


//: PetCount3.java 


// Using Java 1.1 isInstance() 
package ci1.petcount3; 
import java.util.*; 
class Pet {} 
class Dog extends Pet {} 
class Pug extends Dog {} 
class Cat extends Pet {} 
class Rodent extends Pet {} 
class Gerbil extends Rodent {} 
class Hamster extends Rodent {} 
class Counter { int i; } 
public class PetCount3 { 
public static void main(String[] args) { 
Vector pets = new Vector(); 
Class[] petTypes = { 
Pet.class, 


Dog.class, 


Pug.class, 
Cat.class, 
Rodent.class, 
Gerbil.class, 
Hamster.class, 
}; 
try { 
for(int i = 0; i < 15; i++) { 
// Offset by one to eliminate Pet.class: 
int rnd = 1 + (int) ( 
Math.random() * (petTypes.length - 1)); 
pets.addElement ( 
petTypes[rnd].newInstance()); 
} 
} catch(InstantiationException e) {} 
catch(IllegalAccessException e) {} 
Hashtable h = new Hashtable(); 
for(int i = 0; i < petTypes.length; i++) 
h.put(petTypes[i].toString(), 
new Counter()); 
for(int i = 0; i < pets.size(); i++) { 
Object o = pets.elementAt(i); 


// Using isInstance to eliminate individual 


// instanceof expressions: 
for (int j = 0; j < petTypes.length; ++j) 
if (petTypes[j].isInstance(o)) { 
String key = petTypes[j].toString(); 


((Counter )h.get(key)).1++; 


} 
for(int i = 0; i < pets.size(); i++) 
System.out.printin( 
pets.elementAt(i).getClass().toString()); 
Enumeration keys = h.keys(); 
while(keys.hasMoreElements()) { 
String nm = (String)keys.nextElement(); 
Counter cnt = (Counter)h.get(nm); 
System. out.printin( 
nm.substring(nm.lastIndexOf('.') + 1) + 


" quantity: " + cnt.i); 


} 
LW 


可 以 看 到 ，Java ”1.1 的 isInstance() 方 法 已 取消 了 对 instanceof 表 达 式 的 需 
要 。 此 外 ， 这 也 意味 着 一 旦 要 求 添加 新 类 型 宠物 ， 只 需 简 单 地 改变 
petTypes 数 组 即 可 ; 红 需 改动 程序 剩余 的 部 分 〈 但 在 使 用 instanceof 时 却 





是 必需 的 ) 。 
11.2 RTTI 语 法 


Java 用 Class 对 象 实 现 自己 的 RTTI 功 能 即便 我 们 要 做 的 只 是 象 造型 那 
样 的 一 些 工 作 。Class 类 也 提供 了 其 他 大 量 方式 ， 以 方便 我 们 使 用 
RTTI. 


HELIREA Classe RAY AS. BL AR ER AN EN AB, 

一 个 办 法 是 用 一 个 字 串 以 及 Class.forName() 方 法 。 这 是 非常 方便 的 ， 
为 不 需要 那 种 类 型 的 一 个 对 象 来 获取 Class 句 柄 。 然 而 ， 对 于 目 己 感 兴 趣 
的 类 型 ， 如 果 已 有 了 它 的 一 个 对 象 ， 那 么 为 了 取得 Class 句 柄 ， 可 调用 属 
于 Object 根 类 一 部 分 的 一 个 方法 : getClass()。 它 的 作用 是 返回 一 个 特定 
的 Class 句 柄 ， 用 来 表示 对 象 的 实际 类 型 。Class 提 供 了 几 个 有 趣 且 较为 

有 用 的 方法 ， 从 下 例 即 可 看 出 : 


//: ToyTest.java 





// Testing class Class 
interface HasBatteries {} 
interface Waterproof {} 
interface ShootsThings {} 
class Toy { 
// Comment out the following default 
// constructor to see 
// NoSuchMethodError from (*1*) 
Toy() {} 
Toy(int i) {} 
} 


class FancyToy extends Toy 


implements HasBatteries, 
Waterproof, ShootsThings { 
FancyToy() { super(1); } 
} 
public class ToyTest { 
public static void main(String[] args) { 
Class c = null; 
try { 
c = Class.forName("FancyToy"); 
} catch(ClassNotFoundException e) {} 
printInfo(c); 
Class[] faces = c.getInterfaces(); 
for(int i = 0; i < faces.length; i++) 
printInfo(faces[i]); 


Class cy = c.getSuperclass(); 


Object o null; 


try { 
// Requires default constructor: 
o = cy.newInstance(); // (*1*) 

} catch(InstantiationException e) {} 
catch(IllegalAccessException e) {} 


printiInfo(o.getClass()); 


static void printInfo(Class cc) { 
System. out.printin( 
"Class name: " + cc.getName() + 
"is interface? [" + 
cc.isInterface() + "]"); 


} 
Lif] l= 


从 中 可 以 看 出 ，class FancyToy 相 当 复 杂 ， 因 为 它 从 Toy 中 继承 ， 并 实现 
了 HasBatteries，Waterproof 以 及 ShootsThings 的 接口 。 在 main0 中 创建 了 
一 个 Class 人 句柄 ， 并 用 位 于 相应 try 块 内 的 forName() 初 始 化 成 FancyToy。 


Class.getInterfaces 方 法 会 返回 Class 对 象 的 一 个 数组 ， 用 于 表示 包含 在 
Class 对 象 内 的 接口 。 


若 有 一 个 Class 对 象 ， 也 可 以 用 getSuperclassO) 查 询 该 对 象 的 直接 基础 类 
是 什么 。 当 然 ， 这 种 做 会 返回 一 个 Class 句 顶 ， 可 用 它 作 进一步 的 查询 。 
这 意味 着 在 运行 期 的 时 候 ， 完 全 有 机 会 调查 到 对 象 的 完整 层次 结构 。 


若 从 表面 看 ，Class 的 newInstance() 方 法 似乎 是 元 隆 (clone()) 一 个 对 象 
的 男 一 种 手段 。 但 两 者 是 有 区 别 的 。 利 用 newInstance()， 我 们 可 在 没有 
现成 对 象 供 * 殉 隆 ” 的 情况 下 新 建 一 个 对 象 。 就 象 上 面 的 程序 演示 的 那 
样 ， 当 时 没有 Toy 对 象 ， 只 有 cy 即 y 的 Class 对 象 的 一 个 句柄 。 利 用 
它 可 以 实现 “虚拟 构建 问 >”。 换 言 之 ， 我 们 表达 : “尽管 我 不 知道 你 的 准 
确 类 型 是 什么 ， 但 请 你 无 论 如 何 都 正确 地 创建 自己 。” 在 上 述 例子 中 ， 
cy 只 是 一 个 Class 人 句柄 ， 编 译 期 间 并 不 知道 进一步 的 类 型 信息 。 一 旦 新 建 
了 一 个 实例 后 ， 可 以 得 到 Object 句柄 。 但 那个 句柄 指 同 一 个 Toy 对 象 。 
当然 ， 如 果 要 将 除 Object 能 够 接收 的 其 他 任何 消息 发 出 去 ， 首 先 必 须 进 
行 一 些 调查 研究 ， 再 进行 造型 。 除 此 以 外 ， 用 newInstance() 创 建 的 类 必 
须 有 一 个 默认 构建 器 。 没 有 办 法 用 newInstance0O 创 建 拥 有 非 默 认 构 建 嚣 
的 对 象 ， 所 以 在 Java ”1.0 中 可 能 存在 一 些 限制 。 然 而 ，Java ”1.1 的 “ 反 
WAPI CF ERE) 却 允 许 我 们 动态 地 使 用 类 里 的 任何 构建 器 。 





























程序 中 的 最 后 一 个 方法 是 printmnfo0， 它 取得 一 个 Class 句 柄 ， 通 过 
getName(0 获 得 它 的 名 字 ， 并 用 interface0O 调 查 它 是 不 是 一 个 接口 。 


该 程序 的 输出 如 下 : 


Class name: FancyToy is interface? [false] 








Class name: HasBatteries is interface? [true] 
Class name: Waterproof is interface? [true] 
Class name: ShootsThings is interface? [true] 


Class name: Toy is interface? [false] 


所 以 利用 Class 对 象 ， 我 们 几乎 能 将 一 个 对 象 的 祖宗 十 八代 都 调查 出 来 。 
11.3 反射 : 运行 期 类 信息 


如 果 不 知道 一 个 对 象 的 准确 类 型 ，RTTI 会 帮助 我 们 调查 。 但 却 有 一 个 
限制 : 类 型 必须 是 在 编译 期 间 已 知 的， 否则 就 不 能 用 RTTI 调 查 它 ， 进 
而 无 法 展开 下 一 步 的 工作 。 换 言 之 ， 编 译 器 必须 明确 知道 RTTI 要 处 理 
的 所 有 类 。 


从 表面 看 ， 这 似乎 并 不 是 一 个 很 大 的 限制 ， 但 假 知 得 到 的 是 一 个 不 在 目 
己 程 序 空间 内 的 对 象 的 句柄 ， 这 时 又 会 怎样 呢 ? 事实 上 ， 对 象 的 类 即使 
在 编译 期 间 也 不 可 由 我 们 的 程序 使 用 。 例 如 ， 假 设 我 们 从 磁盘 或 者 网 络 
获得 一 系列 字 市 ， 而 且 被 告知 那些 字 节 代表 一 个 类 。 由 于 编译 絮 在 编译 
代码 时 并 不 知道 那个 类 的 情况 ， 所 以 怎样 才能 顺利 地 使 用 这 个 类 呢 ? 


在 传统 的 程序 设计 环境 中 ， 出 现 这 种 情况 的 概率 或 许 很 小 。 但 当 我 们 转 
移 到 一 个 规模 更 大 的 编程 世界 中 ， 却 必须 对 这 个 问题 加 以 高 度 重视 。 第 
一 个 要 注意 的 是 基于 组 件 的 程序 设计 。 在 这 种 环境 下 ， 我 们 用 “快速 应 
HFR” (RAD) 模型 来 构建 程序 项 目 。RAD 一 般 古 在 应 用 程序 构建 工 
其 中 内 建 的 。 这 是 编制 程序 的 一 种 可 视 途 径 〈 在 屏幕 上 以 窗 体 的 形式 出 
现 ) 。 可 将 代表 不 同 组 件 的 图 标 拖 点 到 窗 体 中 。 随 后 ， 通 过 设 定 这 些 组 
件 的 属性 或 者 值 ， 进 行 正确 的 配置 。 设 计 期 间 的 配置 要 求 任何 组 件 都 是 
可 以 “ 例 示 ?的 《〈 即 可 以 自由 获得 它们 的 实例 ) 。 这 些 组 件 也 要 揭示 出 自 














己 的 一 部 分 内 容 ， 人 允许 程序 员 读 取 和 设置 各 种 值 。 此 外 ， 用 于 控制 GUI 
事件 的 组 件 必须 揭示 出 与 相应 的 方法 有 关 的 信息 ， 以 便 RAD 环 境 帮 助 程 
序 员 用 自己 的 代码 宪 亲 这些 由 事件 驱动 的 方法 。 “反射 提供 了 一 种 特殊 
的 机 制 ， 可 以 侦 测 可 用 的 方法 ， 并 产生 方法 名 。 通 过 Java Beans (5613 
Bee eel PA), Java 1.1 为 这 种 基于 组 件 的 程序 设计 提供 了 一 个 基础 


在 运行 期 查询 类 信息 的 另 一 个 原动力 是 通过 网 络 创建 与 执行 位 于 远程 系 
统 上 的 对 象 。 这 就 叫 作 “远程 方法 调用 ”(RMI) ， 它 允许 Java 程 序 〈 版 
本 1.1 以 上 ) 使 用 由 多 人 台 机 器 发 布 或 分 布 的 对 象 。 这 种 对 象 的 分 布 可 能 
可 能 要 做 一 件 计 算 密集 型 的 工作 ， 想 对 它 进 
分 割 ， 让 处 于 空间 状态 的 其 他 机 器 分 担 部 分 工作 ， 从 而 加 快 处 理 进 
Ie. 某 些 情况 下 ， 可 能 需要 将 用 于 控制 特定 类 型 任务 (比如 多 层 客户 / 
服务 器 架构 中 的 “运作 规则 ?) 的 代码 放置 在 一 人 台 特 殊 的 机 器 上 ， 使 这 合 
机 器 成 为 对 那些 行动 进行 描述 的 一 个 通用 储藏 万 。 而 且 可 以 方便 地 修改 
这 个 场所 ， 使 其 对 系统 内 的 所 有 方面 产生 影响 〈 这 古 一 种 特别 有 用 的 设 
计 思 路 ， 因 为 机 器 是 独立 存在 的 ， 所 以 能 轻易 修改 软件 ! ) 。 分 布 式 计 
算 也 能 更 充分 地 发 挥 茶 些 专用 硬件 的 作用 ， 它 们 特别 擅长 执行 一 些 特 定 
的 任务 一 一 例如 秆 阵地 加 一 一 但 对 常规 编程 来 说 却 显得 太 奔 张 或 者 太 品 
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在 Java 1.1 中 ，Class 类 (本 章 前 面 已 有 详细 论述 ) 得 到 了 扩展 ， 可 以 文 
持 “ 反 射 > 的 概念 。 针 对 Field，Method 以 及 Constructor 类 〈 每 个 都 实现 了 
Memberinterface 成 员 接口 ) ， 它 们 都 新 增 了 一 个 库 : 
java.lang.reflect。 这 些 类 型 的 对 象 都 是 JVM 在 运行 期 创建 的 ， 用 于 代表 
未 知 类 里 对 应 的 成 员 。 这 样 便 可 用 构建 右 创 建新 对 象 ， 用 get() 和 set() 方 
法 读 取 和 修改 与 Field 对 象 关联 的 字段 ， 以 及 用 invoke0) 方 法 调用 与 
Method 对 象 关联 的 方法 。 此 外 ， 我 们 可 调用 方法 getFields()， 
getMethods()，getConstructors()， 分 别 返 回 用 于 表示 字段 、 方 法 以 及 构 
建 右 的 对 象 数组 (在 联机 文档 中 ， 还 可 找到 与 Class 类 有 关 的 更 多 的 资 
料 ) 。 因 此 ， 匿 名 对 象 的 类 信息 可 在 运行 期 被 完整 的 揭露 出 来 ， 而 在 编 
译 期 间 不 需要 知道 任何 东西 。 


大 家 要 认识 的 很 重要 的 一 点 是 “反射 ?并 没有 什么 神奇 的 地 方 。 通 过 “ 反 
射 ?同一 个 未 知 类 型 的 对 象 打交道 时 ，JVM 只 是 简单 地 检查 那个 对 象 ， 

并 调查 它 从 属于 哪个 特定 的 类 不 象 以 前 的 RTTI 那 样 )。 但 在 这 之 

后 ， 在 我 们 做 其 他 任何 事情 之 前 ，Class 对 象 必 须 载 入 。 因 此 ， 用 于 那 种 




















特定 类 型 的 .alass 文 件 必 须 能 由 JVM 调 用 〈 要 么 在 本 地 机 器 内 ， 要 么 可 以 
通过 网 络 取 得 ) 。 所 以 RTTI 和 “反射 > 之 间 唯 一 的 区 别 就 是 对 RTTI 来 
说 ， 编 译 器 会 在 编译 期 打开 和 检查 .alass 文 件 。 换 名 话说 ， 我 们 可 以 

用 “普通 ”方式 调用 一 个 对 象 的 所 有 方法 ;但 对 “反射 ?来 说 ，.class 文 件 在 
编译 期 间 是 不 可 使 用 的 ， 而 是 由 运行 期 环境 打开 和 检查 。 


11.3.1 一 个 类 方法 提取 器 


很 少 需要 直接 使 用 反射 工具 ; 之 所 以 在 语言 中 提供 它们 ， 仅 仅 是 为 了 文 
持 其 他 Java 特 性 ， 比 如 对 象 序列 化 (第 10 间 介绍 ) 、Java Beans 以 及 
RMI 本章 后 面 介 绍 ) 。 但 是 ， 我 们 许多 时 候 仍然 需要 动态 提取 与 一 个 
类 有 关 的 资料 。 其 中 特别 有 用 的 工具 便 是 一 个 类 方法 提取 器 。 正 如 前 面 
指出 的 那样 ， 夺 检视 类 定义 源码 或 者 联机 文档 ， 只 能 看 到 在 那个 类 定义 
中 被 定义 或 覆盖 的 方法 ， 基 础 类 那里 还 有 大 量 资料 拿 不 到 。 笠 运 的 

fe, “反射 ?做 到 了 这 一 点 ， 可 用 它 写 一 个 简单 的 工具 ， 令 其 上 自动 展示 整 
个 接口 。 下 面 便 是 具体 的 程序 : 


//: ShowMethods.java 











// Using Java 1.1 reflection to show all the 
// methods of a class, even if the methods are 
// defined in the base class. 
import java.lang.reflect.*; 
public class ShowMethods { 
static final String usage = 
"usage: \n" + 
"ShowMethods qualified.class.name\n" + 
"To show all methods in class or: \n" + 
"ShowMethods qualified.class.name word\n" + 


"To search for methods involving ‘'word'"; 


public static void main(String[] args) { 
if(args.length < 1) { 
System.out.println(usage); 
System.exit(0); 
} 
try { 
Class c = Class.forName(args[0]); 
Method[] m = c.getMethods(); 
Constructor[] ctor = c.getConstructors(); 
if(args.length == 1) { 
for (int 1 = 0; i < m.length; i++) 
System.out.println(m[i].toString()); 
for (int i = 0; i < ctor.length; i++) 
System.out.println(ctor[i].toString()); 
} 
else { 
for (int 1 = 0; i < m.length; i++) 
if(m[i].toString() 
.indexOf(args[1])!= -1) 
System.out.println(m[i].toString()); 
for (int i = 0; i < ctor.length; i++) 
if(ctor[i].toString() 


.indexOf(args[1])!= -1) 


System.out.println(ctor[i].toString()); 
} 
} catch (ClassNotFoundException e) { 


System.out.println("No such class: " + e); 


} 
Lif] l= 


Class 77 i getMethods()#ll getConstructors() P] VAS} Fill i [2] Method Fil 
Constructor 的 一 个 数组 。 每 个 类 都 提供 了 进一步 的 方法 ， 可 解析 出 它们 
所 代表 的 方法 的 名 字 、 人 参数 以 及 返回 值 。 但 也 可 以 象 这 样 一 样 只 使 用 
toString0， 生 成 一 个 含有 完整 方法 签名 的 字 串 。 代 码 剩余 的 部 分 只 是 用 
于 提取 命令 行 信息 ， 判 断 特定 的 签名 是 否 与 我 们 的 目标 字 串 相符 (使 用 
indexOf()) ， 并 打印 出 结果 。 


这 里 便 用 到 了 “反射 ”技术 ， 因 为 由 Class.forName() 产 生 的 结果 不 能 在 编 
译 期 间 获 知 ， 所 以 所 有 方法 签名 信息 都 会 在 运行 期 间 提取 。 千 研究 一 下 
联机 文档 中 关于 “反射 ” (Reflection ) 的 那 部 分 文字 ， 就 会 发 现 它 已 提供 
了 足够 多 的 支持 ， 可 对 一 个 编译 期 完全 未 知 的 对 象 进行 实际 的 设置 以 及 
发 出 方法 调用 。 同 样 地 ， 这 也 属于 几乎 完全 不 用 我 们 操心 的 一 个 步骤 
Java 目 己 会 利用 这 种 文 持 ， 所 以 程序 设计 环境 能 够 控制 Java Beans 
一 一 但 它 无 论 如 何 都 是 非常 有 趣 的 。 


一 个 有 趣 的 试验 是 运行 java ShowMehods ShowMethods。 这 样 做 可 得 到 
一 个 列表 ， 其 中 包括 一 个 public 默 认 构 建 器 ， 斥 管 我 们 在 代码 中 看 见 并 
没有 定义 一 个 构建 器 。 我 们 看 到 的 是 由 编译 器 目 动 合成 的 那 一 个 构建 
器 。 如 果 随 之 将 ShowMethods 设 为 一 个 非 public 类 《〈 即 换 成 “友好 ”类 ) ， 
合成 的 默认 构建 器 便 不 会 在 输出 结果 中 出 现 。 合 成 的 默认 构建 器 会 自动 
获得 与 类 一 样 的 访问 权限 。 


ShowMethods 的 输出 仍然 有 些 “ 不 爽 ”。 例 如 ， 下 面 是 通过 调用 java 
ShowMethods java.lang.String 得 到 的 输出 结果 的 一 部 分 : 








public boolean 


java.lang.String.startswith(java.lang.String, int) 
public boolean 

java.lang.String.startswith(java.lang.String) 
public boolean 


java.lang.String.endsWith(java.lang.String) 


若 能 去 掉 象 java.lang 这 样 的 限定 词 ， 结 果 显 然 会 更 令 人 满意 。 有 鉴于 
此 ， 可 引入 上 一 章 介 绍 的 StreamTokenizer 类 ， 解 决 这 个 问题 : 


//: ShowMethodsClean. java 


// ShowMethods with the qualifiers stripped 

// to make the results easier to read 

import java.lang.reflect.*; 

import java.io.*; 

public class ShowMethodsClean { 

static final String usage = 

"usage: \n" + 
"ShowMethodsClean qualified.class.name\n" + 
"To show all methods in class or: \n" + 
"ShowMethodsClean qualif.class.name word\n" + 
"To search for methods involving ‘word'"; 


public static void main(String[] args) { 


if(args.length < 1) { 
System.out.printlin(usage); 
System.exit(0); 

} 


try { 


Class c = Class.forName(args[0]); 
Method[] m = c.getMethods(); 
Constructor[] ctor = c.getConstructors(); 
// Convert to an array of cleaned Strings: 
String[] n = 
new String[m.length + ctor.length]; 
for(int i = 0; i < m.length; i++) { 
String s = m[i].toString(); 
n[i] = StripQualifiers.strip(s); 
} 
for(int i = 0; i < ctor.length; i++) { 
String s = ctor[i].toString(); 
n[i + m.length] = 
StripQualifiers.strip(s); 
} 
if(args.length == 1) 
for (int 1 = 0; i < n.length; i++) 


System.out.println(n[i]); 


else 
for (int i = 0; i < n.length; i++) 
if(n[i].indexOf(args[1])!= -1) 
System.out.println(n[i]); 
} catch (ClassNotFoundException e) { 


System.out.printin("No such class: " + e); 


} 


class StripQualifiers { 
private StreamTokenizer st; 
public StripQualifiers(String qualified) { 
st = new StreamTokenizer ( 
new StringReader (qualified) ); 
st.ordinaryChar(' '); // Keep the spaces 
} 
public String getNext() { 
String s = null; 
try { 
if(st.nextToken() != 
StreamTokenizer.TT_EOF) { 
Switch(st.ttype) { 


case StreamTokenizer.TT_EOL: 


s = null; 
break; 
case StreamTokenizer.TT_NUMBER: 
s = Double.toString(st.nval); 
break; 
case StreamTokenizer.TT_WORD: 
S = new String(st.sval); 
break; 
default: // single character in ttype 


s = String.valueOf((char)st.ttype); 


} 


} catch(IOException e) { 
System.out.println(e); 
} 
return s; 
} 
public static String strip(String qualified) { 
StripQualifiers sq = 
new StripQualifiers(qualified); 
String s = "", si; 
while((si = sq.getNext()) != null) { 


int lastDot = si.lastIndexOf('.'); 


if(lastDot != -1) 
Si = si.substring(lastDot + 1); 
s += Sİ; 
} 
return s; 


} 
} ///:~ 





ShowIMethodsClean 方 法 非常 接近 前 一 个 ShowMethods， 只 是 它 取 得 了 
Method 和 Constructor 数 组 ， 并 将 它们 转换 成 单个 String 数 组 。 随 后 ， 
个 这 样 的 String 对 象 都 在 StripQualifiers.StripO0 里 * 过 ”一 遍 ， 删 除 所 有 方法 
限定 词 。 正 如 大 家 看 到 的 那样 ， 此 时 用 到 了 StreamTokenizer 和 String 来 
完成 这 省 工作 


假如 记 不 得 一 个 类 是 否 有 一 个 特定 的 方法 ， 而 且 不 想 在 联机 文档 里 逐步 
检查 类 结构 ， 或 者 不 知道 那个 类 是 否 能 对 茶 个 对 象 〈 如 Color 对 象 ) 做 
某 件 事情 ， 该 工具 便 可 节省 大 量 编程 时 间 。 


第 17 草 提供 了 这 个 程序 的 一 个 GUI 版 本 ， 可 在 上 自己 写 代码 的 时 候 运行 
它 ， 以 便 快 速 碍 找 需要 的 东西 。 


11.4 总 结 


利用 RTTI 可 根据 一 个 匿名 的 基础 类 句柄 调查 出 类 型 信息 。 但 正 是 由 于 

这 个 原因 ， 新 手 们 极 易 误 用 它 ， 因 为 有 些 时 候 多 形 性 方法 便 足 够 了 。 对 
那些 以 前 习惯 程序 化 编程 的 人 来 说 ， 极 易 将 他 们 的 程序 组 织 成 一 系列 

switch 语 句 。 他 们 可 能 用 RTTI 做 到 这 一 点 ， 从 而 在 代码 开发 和 维护 中 损 
失 多 形 性 拉 术 的 重要 价值 。Java 的 要 求 是 让 我 们 尽 可 能 地 采用 多 形 性 ， 

只 有 在 极 特别 的 情况 下 才 使 用 RTTI。 


但 为 了 利用 多 形 性 ， 要 求 我 们 拥有 对 基础 类 定义 的 控制 权 ， 因 为 有 些 时 
候 在 程序 范围 之 内 ， 可 能 发 现 基 础 类 并 未 包括 我 们 想 要 的 方法 。 知 基础 
类 来 目 一 个 库 ， 或 者 由 别 的 什么 东西 控制 着 ，RTTI 便 是 一 种 很 好 的 解 














ATR: 可 继承 一 个 新 类 型 ， 然 后 添加 自己 的 额外 方法 。 在 代码 的 其 他 
地 方 ， 可 以 侦 测 自己 的 特定 类 型 ， 并 调用 那个 特殊 的 方法 。 这 样 做 不 会 
破坏 多 形 性 以 及 程序 的 扩展 能 力 ， 因 为 新 类 型 的 添加 不 要 求人 宜 找 程序 中 
的 switch 语 句 。 但 在 需要 新 特性 的 主体 中 添加 新 代码 时 ， 就 必须 用 RTTI 
侦 测 自己 特定 的 类 型 。 


从 某 个 特定 类 的 利益 的 角度 出 发 ， 在 基础 类 里 加 入 一 个 特性 后 ， 可 能 意 
味 着 从 那个 基础 类 衍生 的 其 他 所 有 类 都 必须 获得 一 些 无 意义 的 “鸡肋 ”。 
这 使 得 接口 变 得 含义 模糊 。 特 有 人 从 那个 基础 类 继承 ， 且 必须 履 善 抽象 
方法 ， 这 一 现象 便 会 使 他 们 陷入 困扰 。 比 如 现在 用 一 个 类 结构 来 表示 乐 
器 〈Instrument) 。 假 定 我 们 想 清洁 管弦 乐队 中 所 有 适当 乐器 的 通气 音 
ke (Spit Valve) ， 此 时 的 一 个 办 法 是 在 基础 类 Instrument 中 置 入 一 个 
ClearSpitValve() 方 法 。 但 这 样 做 会 造成 一 个 误区 ， 因 为 它 上 暗示 着 打击 乐 
器 和 电子 乐器 中 也 有 首 栓 。 针 对 这 种 情况 ，RTTI 提 供 了 一 个 更 合理 的 
解决 方案 ， 可 将 方法 置 入 特定 的 类 中 (此 时 是 Wind， 即 “通气 口 ”) 一 一 
这 样 做 是 可 行 的。 但 事实 上 一 种 更 合理 的 方案 是 将 prepareInstrument() 置 
人 
RTTI. 


最 后 ，RTTI 有 时 能 解决 效率 问题 。 若 代码 大 量 运用 了 多 形 性 ， 但 其 中 
的 一 个 对 象 在 执行 效率 上 很 有 问题 ， 便 可 用 RTTI 找 出 那个 类 型 ， 然 后 
写 一 段 适当 的 代码 ， 改 进 其 效率 。 

11.5 练习 


和 
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(2) ”在 ToyTest.java 中 ， 将 Toy 的 默认 构建 器 标记 成 注释 信息 ， 解 释 随 之 
发 生 的 事情 。 


(3) 新 建 一 种 类 型 的 集合 ， 令 其 使 用 一 个 Vector。 捕 获 置 入 其 中 的 第 一 个 
对 象 的 类 型 ， 然 后 从 那 时 起 只 允许 用 户 插入 那 种 类 型 的 对 象 。 


人 
J 对象。 


(5) 根据 本 章 的 说 明 ， 实 现 clearSpitValve()。 





(6) 实现 本 章 介 绍 的 rotate(Shape) 方 法 ， 令 其 检查 是 否 已 经 旋转 了 一 个 圆 
(知己 旋转， 就 不 再 执行 旋转 操作 ) o 





第 12 章 传递 和 返回 对 象 


到 目前 为 止 ， 读 者 应 对 对 象 的 “传递 "有 了 一 个 较为 深刻 的 认识 ， 记 住 实 
际 传 递 的 只 是 一 个 句柄 。 


在 许多 程序 设计 语言 中 ， 我 们 可 用 语言 的 “普通 ”方式 到 处 传递 对 象 ， 而 
且 大 多 数 时 候 都 不 会 遇 到 问题 。 但 有 些 时 候 却 不 得 不 采取 一 些 非 党 做 
法 ， 使 得 情况 突然 变 得 稍微 复杂 起 来 〈 在 C++ 中 则 是 变 得 非常 复杂 ) 。 
Java 亦 不 例外 ， 我 们 十 分 有 必要 准确 认识 在 对 象 传递 和 赋值 时 所 发 生 的 
一 切 。 这 正 是 本 章 的 宗旨 。 


右 读 者 是 从 菏 些 特殊 的 程序 设计 环境 中 转移 过 来 的 ， 那 么 一 般 都 会 问 
到 : “Java 有 指针 吗 ? "有些 人 认为 指针 的 操作 很 困难 ， 而 且 十 分 危险 ， 
所 以 一 厢 情 愿 地 认为 它 没有 好 处 。 同 时 由 于 Java 有 如 此 好 的 口碑 ， 所 以 
应 该 很 轻易 地 免除 目 己 以 前 编程 中 的 麻烦 ， 其 中 不 可 能 夹带 有 指针 这 样 
的 “危险 品 ”。 然 而 准确 地 说 ，Java 是 有 指针 的 ! 事实 上 ，Java 中 每 个 对 
象 〈 除 基本 数据 类 型 以 外 ) 的 标识 符 部 属于 指针 的 一 种 。 但 它们 的 使 用 
受到 了 严格 的 限制 和 防范 ， 不 仅 编译 器 对 它们 有 "和 戒心 ”， 运 行 期 系统 也 
不 例外 。 或 者 换 从 另 一 个 角度 说 ，Java 有 指针 ， 但 没有 传统 指针 的 麻 
烦 。 我 曾 一 度 将 这 种 指针 叫做 “句柄 ”， 但 你 可 以 把 它 想 像 成 < 安全 指 
针 ”。 和 预备 学 校 为 学 生 提 供 的 安全 甬 刀 类 似 一 一 除非 特别 有 意 ， 否 则 
不 会 伤 着 自己， 只 不 过 有 时 要 慢 慢 来 ， 要 习惯 一 些 沉 问 的 工作 。 



































12.1 传递 句柄 


将 句柄 传递 进入 一 个 方法 时 ， 指 向 的 仍然 是 相同 的 对 象 。 一 个 简单 的 实 
这 个 程序 时 有 了 麻烦， 请 参考 第 eo 
aE cee 2) 


//: PassHandles.java 





// Passing handles around 
package c12; 
public class PassHandles { 
static void f(PassHandles h) { 
System.out.printin("h inside f(): " + h); 
} 
public static void main(String[] args) { 
PassHandles p = new PassHandles(); 
System.out.println("p inside main(): " + p); 
f(p); 
} 
} ///:~ 


toString 方 法 会 在 打印 语句 里 自动 调用 ， 而 PassHandles 直 接 从 Object 继 

承 ， 没 有 toString 的 重新 定义 。 因 此 ， 这 里 会 采用 toString 的 Object 版 本 ， 

打印 出 对 象 的 类 ， 接 着 是 那个 对 象 所 在 的 位 置 (不 是 句柄 ， 而 是 对 象 的 
实际 存储 位 置 ) 。 输 出 结果 如 下 : 


p inside main(): PassHandles@1653748 


h inside f() : PassHandles@1653748 


可 以 看 到 ， 无 论 p 还 是 h 引 用 的 都 是 同一 个 对 象 。 这 比 复制 一 个 新 的 
PassHandles 对 象 有 效 多 了 ， 使 我 们 能 将 一 个 参数 发 给 一 个 方法 。 但 这 样 
做 也 带 来 了 另 一 个 重要 的 问题 。 


12.1.1 别名 问题 

“别名 ”意味 着 多 个 句柄 都 试图 指向 同一 个 对 象 ， 就 象 前 面 的 例子 展示 的 
那样 。 知 有 人 回 那 个 对 象 里 写 入 一 点 什么 东西 ， 就 会 产生 别名 问题 。 知 
其 他 句柄 的 所 有 者 不 希望 那个 对 象 改 变 ， 恐 介 束 要 失望 了 。 这 可 用 下 面 
这 个 简单 的 例子 说 明 : 


//: Aliasi.java 








// Aliasing two handles to one object 
public class Aliasi { 
int 1; 
Aliasi(int ii) { i = ii; } 
public static void main(String[] args) { 
Alias1 x = new Alias1(7); 
Alias1 y = x; // Assign the handle 
System.out.println("x: " + x.i); 
System.out.printin("y: " + y.i); 
System.out.println("Incrementing x"); 
x.i++; 
System.out.println("x: " + x.i); 


System.out.println("y: " + y.i); 


} ///:~ 


对 下 面 这 行 : 

Aliasl y = x; // Assign the handle 

它 会 新 建 一 个 Alias1 人 句柄， 但 不 是 把 它 分 配给 由 new 创 建 的 一 个 新 鲜 对 
象 ， 而 是 分 配给 一 个 现 有 的 句柄 。 所 以 句柄 x 的 内 容 一 一 即 对 象 z 指 回 的 


地 址 一 一 被 分 配给 y， 所 以 无 论 x 还 是 y 都 与 相同 的 对 象 连接 起 来 。 这 样 
一 来 ， 一 旦 x 的 i 在 下 述 语句 中 增值 : 





义 .i 十 十 ; 


y 的 i 值 也 必然 受到 影响 。 从 最 终 的 输出 就 可 以 看 出 : 


xX: 7 


y: 7 
Incrementing x 
xX: 8 


y: 8 








SECS te EL Be HI 1S A DTS a eT WEAN IRL: 不 要 有 意 将 多 个 句柄 指 
问 同一 个 作用 域内 的 同一 个 对 象 。 这 样 做 可 使 代码 更 易 理解 和 调试 。 然 
而 ， 一 旦 准备 将 句柄 作为 一 个 目 变 量 或 参数 传递 一 一 这 是 Java 设 想 的 正 
常 方 法 一 一 别名 问题 就 会 自动 出 现 ， 因 为 创建 的 本 地 句柄 可 能 修改 “外 
部 对 象 ” (在 方法 作用 域 之 外 创建 的 对 象 》。 下 面 是 一 个 例子 : 


//: Alias2.java 





// Method calls implicitly alias their 


// arguments. 


public class Alias2 { 
int 1; 
Alias2(int ii) { i = ii; } 
static void f(Alias2 handle) { 
handle.it++; 
} 
public static void main(String[] args) { 
Alias2 x = new Alias2(7); 
System.out.printin("x: " + x.i); 
System.out.println("Calling f(x)"); 
f(x); 
System.out.println("x: " + x.i); 
} 
EJL a~ 


输出 如 下 : 
x: 7 
Calling f(x) 
x: 8 


方法 改变 了 目 己 的 参数 一 一 外 部 对 象 。 一 旦 遇 到 这 种 情况 ， 必 须 判 断 它 
是 否 合 理 ， 用 户 是 否 愿 意 这 样 ， 以 及 是 不 是 会 造成 问题 。 


通常 ， 我 们 调用 一 个 方法 是 为 了 产生 人 返回 值 ， 或 者 用 它 改 变 为 其 调用 方 
法 的 那个 对 象 的 状态 (方法 其 实 就 是 我 们 癌 那 个 对 象 “及 一 条 消 恩 ”的 方 
式 ) 。 很 少 需要 调用 一 个 方法 来 处 理 它 的 参数 ;这 叫 作 利用 方法 的 “ 副 


作用 ”(Side Effect) 。 所 以 倘 硝 创建 一 个 会 修改 自己 参数 的 方法 ， 必 须 
加 用户 明确 地 指出 这 一 情况 ， 并 警告 使 用 那个 方法 可 能 会 有 的 后 果 以 及 
a erry rae a Wh mem aera es 





Ani FE PAE Vd EMTS, AAT HEROES, wD 
在 自己 的 方法 内 部 制作 一 个 副本 ， 从 而 保护 那个 参数 。 本 章 的 大 多 数 内 
容 都 是 围绕 这 个 问题 展开 的 。 


12.2 制作 本 地 副本 


稍微 总 结 一 下 : Java 中 的 所 有 目 变 量 或 参数 传递 都 是 通过 传递 句柄 进行 
的 。 也 就 是 说 ， 当 我 们 传递 “一 个 对 象 ? 时 ， 实 际 传 递 的 只 是 指 同 位 于 方 
法 外 部 的 那个 对 象 的 “一 个 句柄 ”。 所 以 一 旦 要 对 那个 句柄 进行 任何 修 
改 ， 便 相当 于 修改 外 部 对 象 。 此 外 : 

四 参数 传递 过 程 中 会 目 动产 生 别 名 问题 

@ 个 存在 本 地 对 象 ， 只 有 本 地 人 句柄 

eM A CITE, MARA 

my Ay Fe EMT Te)” E Java EA eT Al ell 


加 没有 语言 上 的 文 持 (如 常量 ) 可 防止 对 象 被 修改 (以 避免 别名 的 副 作 
用 ) 











行 只 是 从 对 象 中 读 取 信息 ， 而 不 修改 它 ， 传 递 句柄 便 是 自 变量 传递 中 最 
有 效 的 一 种 形式 。 这 种 做 非常 恰当 ; 默认 的 方法 一 般 也 是 最 有 效 的 方 
法 。 然 而 ， 有 时 仍 需 将 对 象 当 作 * 本 地 的 ?对 待 ， 使 我 们 作出 的 改变 只 影 
啊 一 个 本 地 副本 ， 不 会 对 外 面 的 对 象 造成 影响 。 许 多 程序 设计 语言 都 文 
持 在 方法 内 自动 生成 外 部 对 象 的 一 个 本 地 副本 《注释 四) 。 尽 管 Java 不 
具备 这 种 能 力 ， 但 允许 我 们 达到 同样 的 效果 。 


©: 在 C 语 言 中 ， 通 常 控制 的 是 少量 数据 位 ， 默 认 操 作 是 按 值 传递 。 
C++ 也 必须 着 照 这 一 形式 ， 但 按 值 传递 对 象 并 非 肯 定 是 一 种 有 效 的 方 
人 
痛 的 事情 。 


12.2.1 按 值 传递 


首先 要 解决 术语 的 问题 ， 最 适合 “ 按 值 传递 "的 看 起 来 是 自 变 量 。“ 按 值 
传递 ”以 及 它 的 含义 取决 于 如 何 理 解 程序 的 运行 方式 。 最 第 见 的 意思 古 
获得 要 传递 的 任何 东西 的 一 个 本 地 副本 ,但 这 里 真正 的 问题 是 如 何 看 待 
目 己 准备 传递 的 东西 。 对 于 “ 按 值 传递 ”的 含义 ， 目 前 存在 两 种 存在 明显 
区 别 的 见解 : 


























(1) Java 按 值 传递 任何 东西 。 知 将 基本 数据 类 型 传递 进入 一 个 方法 ， 会 明 
确 得 到 基本 数据 类 型 的 一 个 副本 。 但 知 将 一 个 句柄 传递 进入 方法 ， 得 到 
的 是 句柄 的 副本 。 所 以 人 们 认为 一切” 都 按 值 传递 。 当 然 ， 这 种 说 法 也 
有 一 个 前 提 : 句柄 肯定 也 会 被 传递 。 但 Java 的 设计 方案 似乎 有 些 超前 ， 

允许 我 们 忽略 (大 多 数 时 候 〉 自己 处 理 的 是 一 个 句柄 。 也 就 是 说 ， 它 人 允 
许 我 们 将 句柄 假想 成 对象”， 因 为 在 发 出 方法 调用 时 ， 系 统 会 目 动 照管 
两 者 间 的 差异 。 


(2) Java 主 要 按 值 传递 (无 自 变 量 ) ， 但 对 象 却 是 按 引 用 传递 的 。 得 到 这 
个 结论 的 前 提 是 句柄 只 是 对 象 的 一 个 “别名 ”， 所 以 不 考虑 传递 句柄 的 问 
题 ， 而 是 直接 指出 “我 准备 传递 对 象 "。 由 于 将 其 传递 进入 一 个 方法 时 没 
有 获得 对 象 的 一 个 本 地 副本 ， 所 以 对 象 显然 不 是 按 值 传递 的 。Sun 公 司 
似乎 在 菜 种 程度 上 文 持 这 一 见解 ， 因 为 它 * 保 留 但 未 实现 ”的 关键 字 之 一 
便 是 byvalue《〈 按 值 ) 。 但 没 人 知道 那个 关键 字 什么 时 候 可 以 发 挥 作用 。 


尽管 存在 两 种 不 同 的 见解 ， 但 其 间 的 分 虑 归根 到 底 是 由 于 对 “句柄 ”的 不 
同 解释 造成 的 。 我 打算 在 本 书 剩 下 的 部 分 里 回避 这 个 问题 。 大 家 不 久 囊 
会 知道 ， 这 个 问题 争论 下 去 其 实 是 没有 意义 的 一 一 最 重要 的 是 理解 一 个 
句柄 的 传递 会 使 调用 者 的 对 象 发 生意 外 的 改变 。 


12.2.2 WER Z 


Ar ii (ECL — POT RR, TAN ANA AR a eT AR, HI PE AT RAN — 
个 本 地 副本 。 这 也 是 本 地 副本 最 常见 的 一 种 用 途 。 寿 决定 制作 一 个 本 地 
副本 ， 只 需 简 单 地 使 用 clone() 方 法 即 可 。Clone 是 “克隆 ”的 意思 ， 即 制作 
完全 一 模 一 样 的 副本 。 这 个 方法 在 基础 类 Object 中 定义 

成 “protected”( 受 保护 ) 模式 。 但 在 希望 殉 隆 的 任何 衍生 类 中 ， 必 须 将 

HA mAN “public iN. Pi, PREEK Vector ti Sclone(), Pre REN 

Vector 调用 clone0， 如 下 所 示 : 


























//: Cloning.java 


// The clone() operation works for only a few 
// items in the standard Java library. 


import java.util.*; 


class Int { 
private int i; 
public Int(int ii) { i = ii; } 
public void increment() { i++; } 
public String toString() { 


return Integer.toString(i); 


} 


public class Cloning { 
public static void main(String[] args) { 
Vector v = new Vector(); 
for(int i = 0; i < 10; i++ ) 
v.addElement(new Int(i)); 
System.out.println("v: " + v); 
Vector v2 = (Vector)v.clone(); 
// Increment all v2's elements: 
for(Enumeration e = v2.elements(); 
e.hasMoreElements(); ) 
((Int)e.nextElement()).increment(); 
// See if it changed v's elements: 


System.out.printin("v: " + v); 


de 





clone() 方 法 产生 了 一 个 Object， 后 者 必须 立即 重新 造型 为 正确 类 型 。 这 
个 例子 指出 Vector 的 clone() 方 法 不 能 自动 尝试 克隆 Vector 内 包含 的 每 个 
对 象 由 于 别名 问题 ， 老 的 Vector 和 克隆 的 Vector 都 包含 了 相同 的 对 
象 。 我 们 通常 把 这 种 情况 叫 作 “简单 复制 ”或 者 “ 浅 层 复制 ”， 因 为 它 只 复 
制 了 一 个 对 象 的 “表面 * 部 分 。 实 际 对 象 除 包含 这 个 “表面 ”以 外 ， 还 包括 
句 顶 指 癌 的 所 有 对 象 ， 以 及 那些 对 象 叉 指 同 的 其 他 所 有 对 象 ， 由 此 类 
推 。 这 便 是 “对 象 网 ?或 “对 象 关系 网 ”的 由 来 。 知 能 复制 下 所 有 这 张 网 ， 
便 叫 作 * 全 面 复制 ?或 者 “深层 复制 ”。 


在 输出 中 可 看 到 浅 层 复 制 的 结果 ， 注 意 对 v2 采 取 的 行动 也 会 影响 到 V: 


v: [0, 1, 2, 3, 4, 5, 6, 7, 8, 9] 








v: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10] 


一 般 来 说 ， 由 于 不 了 敢 保证 Vector 里 包含 的 对 象 是 “可 以 克隆 ”( 注 释 @)) 
的 ， 所 以 最 好 不 要 试图 克隆 那些 对 象 。 


@:“ 可 以 克隆 ”用 英语 讲 是 cloneable， 请 留意 Java 库 中 专门 保留 了 这 样 
的 一 个 关键 字 。 


12.2.3 使 类 具有 克隆 能 


尽管 克隆 方法 是 在 所 有 类 最 基本 的 Object 中 定义 的 ， 但 克隆 仍然 不 会 在 
每 个 类 里 自动 进行 。 这 似乎 有 些 不 可 思议 ， 因 为 基础 类 方法 在 衍生 类 里 
是 肯定 能 用 的 。 但 Java 确 实 有 点 儿 反 其 道 而 行 之 ; 如 果 想 在 一 个 类 里 使 
用 克隆 方法 ， 唯 一 的 办 法 就 是 专门 添加 一 些 代码 ， 以 便 保证 元 隆 的 正常 
进行 5 

1. 使 用 protected 时 的 技巧 

为 避免 我 们 创建 的 每 个 类 都 默认 具有 元 隆 能 力 ，clone() 方 法 在 基础 类 
Object 里 得 到 了 “保留 ”〈 设 为 protected) 。 这 样 造成 的 后 果 就 是 : 对 那 


些 简单 地 使 用 一 下 这 个 类 的 客户 程序 员 来 说 ， 他 们 不 会 默认 地 拥有 这 个 
方法 ; 其 次 ， 我 们 不 能 利用 指 同 基础 类 的 一 个 句柄 来 调用 cloneO0 〈 尽 管 




















那样 做 在 某 些 情况 下 特别 有 用 ， 比 如 用 多 形 性 的 方式 克隆 一 系列 对 
象 ) 。 在 编译 期 的 时 候 ， 这 实际 是 通知 我 们 对 象 不 可 克隆 的 一 种 方式 
一 一 而 且 最 奇怪 的 是 ，Java 库 中 的 大 多 数 类 都 不 能 克隆 。 因 此 ， 假 如 我 
们 执行 下 述 代码 : 








Integer x = new Integer(I); 
x = x.clone(); 


那么 在 编译 期 ， 就 有 一 条 讨厌 的 错误 消息 弹出 ， 告 诉 我 们 不 可 访问 
clone() Alnteger#FiXA Bi 6, MH EX protected KK tte BK 
ABa 


但 是 ， 假 若 我 们 是 在 一 个 从 Object 衍生 出 来 的 类 中 〈 所 有 类 都 是 从 
Object 衍生 的 ) ， 就 有 权 调 用 Object.clone0， 因 为 它 是 “protected”， 而 且 
我 们 在 一 个 继承 器 中 。 基 础 类 clone0 提 供 了 一 个 有 用 的 功能 它 进行 
的 是 对 衍生 类 对 象 的 真正 “ 按 位 ”复制 ， 所 以 相当 于 标准 的 克隆 行动 。 然 
而 ， 我 们 随后 需要 将 自己 的 元 隆 操作 设 为 public， 否 则 无 法 访问 。 总 
之 ， 克 隆 时 要 注意 的 两 个 关键 问题 是 :几乎 肯定 要 调用 super.clone()， 
以 及 注意 将 元 隆 设 为 public。 


有 时 还 想 在 更 深层 的 衔 生 类 中 禾 盖 clone()， 否 则 就 直接 使 用 我 们 的 
clone()〔 现 在 已 成 为 public〉 ， 而 那 并 不 一 定 是 我 们 所 希望 的 (然而 ， 
由 于 Object.cloneO 已 制作 了 实际 对 象 的 一 个 副本 ， 所 以 也 有 可 能 允许 这 
种 情况 ) 。protected 的 技巧 在 这 里 只 能 用 一 次 : 首次 从 一 个 不 具备 克隆 
能 力 的 类 继承 ， 而 且 想 使 一 个 类 变 成 “能 够 克隆”。 而 在 从 我 们 的 类 继承 
的 任何 场合 ，clone() 方 法 都 是 可 以 使 用 的 ， 因 为 Java 不 可 能 在 衍生 之 后 
反而 缩小 方法 的 访问 范围 。 换 言 之 ， 一 旦 对 象 变 得 可 以 克隆 ， 从 它 衍生 
的 任何 东西 都 是 能 够 克隆 的 ， 除 非 使 用 特殊 的 机 制 ( 后 面 讨论 ) 令 

其 “关闭 ”克隆 能 


2. 实现 Cloneable 接 口 


为 使 一 个 对 象 的 克隆 能 力 功 成 圆满 ， 还 需要 做 男 一 件 事情 : 实现 
Cloneable 接 口 。 这 个 接口 使 人 稍 党 奇怪 ， 因 为 它 是 空 的 ! 











interface Cloneable {} 





之 所 以 要 实现 这 个 空 接口 ， 显 然 不 是 因为 我 们 准备 上 漳 造 型 成 一 个 
Cloneable， 以 及 调用 它 的 某 个 方法 。 有 些 人 认为 在 这 里 使 用 接口 属于 一 
种 “欺骗 "行为 ， 因 为 它 使 用 的 特性 打 的 是 别 的 主意 ， 而 非 原 来 的 意思 。 
Cloneable interface 的 实现 扮演 了 一 个 标记 的 角色 ， 封 装 到 类 的 类 型 中 。 


两 方面 的 原因 促成 了 Cloneable interface 的 存在 。 首 先 ， 可 能 有 一 个 上 济 
造型 句柄 指 同 一 个 基础 类 型 ， 而 且 不 知道 它 是 否 真 的 能 克隆 那个 对 象 。 
在 这 种 情况 下 ， 可 用 instanceof 关 键 字 (第 11 间 有 介绍 ) WAM AM 
实 同一 个 能 元 隆 的 对 象 连接 : 








if(myHandle instanceof Cloneable) // ... 

第 二 个 原因 是 考虑 到 我 们 可 能 不 愿 所 有 对 象 类 型 都 能 克隆 。 上 所 以 
Object.clone0 会 验证 一 个 类 是 否 真 的 是 实现 了 Cloneable 接 口 。 知 答案 是 
否定 的 ， 则 “ 括 ? 出 一 个 CloneNotSupportedException 违 例 。 所 以 在 一 般 情 
况 下 ， 我 们 必须 将 “implement Cloneable” 作 为 对 元 隆 能 力 提 供 支 持 的 一 


部 分 。 
12.2.4 成 功 的 元 隆 


理解 了 实现 clone() 方 法 背后 的 所 有 细 市 后 ， 便 可 创建 出 能 方便 复制 的 
类 ， 以 便 提供 了 一 个 本 地 副本 : 


//: LocalCopy.java 








// Creating local copies with clone() 
import java.util.*; 
class MyObject implements Cloneable { 
int 1; 
MyObject(int ii) { i = ii; } 
public Object clone() { 
Object o = null; 


try { 


o = super.clone(); 
} catch (CloneNotSupportedException e) { 
System.out.println("MyObject can't clone"); 
} 
return oO; 
} 
public String toString() { 


return Integer.toString(i); 


public class LocalCopy { 

static MyObject g(MyObject v) { 
// Passing a handle, modifies outside object: 
Ve Let; 
return v; 

} 

static MyObject f(MyObject v) { 
v = (MyObject)v.clone(); // Local copy 
v.i++; 
return vV; 

} 

public static void main(String[] args) { 


MyObject a = new MyObject(11); 


MyObject b = g(a); 
// Testing handle equivalence, 
// not object equivalence: 
if(a == b) 
System.out.println("a == b"); 
else 
System.out.printlin("a != b"); 
System.out.printin("a = " + a); 
System.out.printlin("b = " + b); 
MyObject c = new MyObject(47); 
MyObject d = f(c); 
if(c == d) 
System.out.println("c == d"); 
else 


System.out.printin("c != d"); 


System.out.println("c = " + c); 
System.out.println("d = " + d); 
iF 
} ///:~ 


不 管 怎 样 ，clone() 必 须 能 够 访问 ， 所 以 必须 将 其 设 为 public (公共 
的 ) 。 其 次 ， 作 为 clone() 的 初期 行动 ， 应 调用 clone() 的 基础 类 版 本 。 这 
里 调用 的 clone() 是 Object 内 部 预先 定义 好 的 。 之 所 以 能 调用 它 ， 是 由 于 
它 具 有 protected〈 受 到 保护 的 ) 属性 ， 所 以 能 在 衍生 的 类 里 访问 。 





Object.clone0 会 检查 原先 的 对 象 有 多 大 ， 再 为 新 对 象 腾 出 足够 多 的 内 
存 ， 将 所 有 二 进 制 位 从 原来 的 对 象 复 制 到 新 对 象 。 这 叫 作 * 按 位 复制 ”， 
而 且 按 一 般 的 想法 ， 这 个 工作 应 该 是 由 clone() 方 法 来 做 的 。 但 在 
Object.clone0 正 式 开始 操作 前 ， 首 先 会 检查 一 个 类 是 否 Cloneable， 即 是 
a FA be BEE 换言之 ， 它 是 否 实现 了 Cloneable 接 口 。 知 未 实现 ， 
Object.clone0O 就 据 出 一 个 CloneNotSupportedException 违 例 ， 指 出 我 们 不 
能 克隆 它 。 因 此 ， 我 们 最 好 用 一 个 try-catch 块 将 对 super.cloneO 的 调用 代 
BAH GAR) 起 来 ， 试 图 捕获 一 个 应 当 永 不 出 现 的 违例 〈 因 为 这 里 
确实 已 实现 了 Cloneable 接 口 ) 。 


在 LocalCopy 中 ， 两 个 方法 g0 和 f0 揭 示 出 两 种 参数 传递 方法 间 的 差异 。 
其 中 ，g0 演 示 的 是 按 引 用 传递 ， 它 会 修改 外 部 对 象 ， 并 返回 对 那个 外 部 
对 象 的 一 个 引用 。 而 f0 是 对 自 变 量 进行 克隆 ， 所 以 将 其 分 离 出 来 ， 并 让 
原来 的 对 象 保 持 独立 。 随 后 ， 它 继续 做 它 和 希望 的 事情 。 甚 至 能 返回 指 加 
这 个 新 对 象 的 一 个 句柄 ， 而 且 不 会 对 原来 的 对 象 产生 任何 副作用 。 注 意 
下 面 这 个 多 少 有 些 上 古怪 的 语句 : 


v = (MyObject)v.clone(); 


它 的 作用 正 古 创建 一 个 本 地 副本 。 为 避免 被 这 样 的 一 个 语句 搞 混淆 ， 记 
住 这 种 相当 奇怪 的 编码 形式 在 Java 中 是 完全 允许 的 ， 因 为 有 一 个 名 字 的 
所 有 东西 实际 都 是 一 个 句柄 。 所 以 句柄 y 用 于 克隆 一 个 它 所 指 同 的 副 
本 ， 而 且 最 终 返回 指 网 基础 类 型 Object 的 一 个 句柄 《因为 它 在 
Object.cloneO 中 是 那样 被 定义 的 ) ， 随 后 必须 将 其 造型 为 正确 的 类 型 。 


在 main0 中 ， 两 种 不 同 参数 传递 方式 的 区 别 在 于 它们 分 别 测试 了 一 个 不 
同 的 方法 。 输 出 结果 如 下 : 
































a == b 
a= 12 
b = 12 
c l=d 
c = 47 


大 家 要 记 住 这 样 一 个 事实 : Java 对 “是 否 等 价 ” 的 测试 并 不 对 所 比较 对 象 
的 内 部 进行 检查 ， 从 而 核实 它们 的 值 是 否 相 同 。== 和 != 运 算 符 只 是 简单 
地 对 比 句柄 的 内 容 。 若 句柄 内 的 地 址 相同 ， 就 认为 句柄 指向 同样 的 对 
象 ， 所 以 认为 它们 是 “等 价 ” 的 。 所 以 运算 符 真 正 检 测 的 是 “由 于 别名 问 
题 ， 人 句柄 是 否 指 向 同 一 个 对 象 ? ” 


12.2.5 Object.cloneO 的 效果 


调用 Object.clone0 时 ， 实 际 发 生 的 是 什么 事情 呢 ? 当 我 们 在 自己 的 类 里 
履 盖 cloneO0 时 ， 什 么 东西 对 于 super.clone0) 来 说 是 最 关键 的 呢 ? 根 类 中 的 
clone() 方 法 负责 建立 正确 的 存储 容量 ， 并 通过 “ 按 位 复制 "将 二 进 制 位 从 
原始 对 象 中 复制 到 新 对 象 的 存储 空间 。 也 就 是 说 ， 它 并 不 只 是 预 留存 储 
空间 以 及 复制 一 个 对 象 一 一 实际 需要 调查 出 欲 复制 之 对 象 的 准确 大 小 ， 
然后 复制 那个 对 象 。 由 于 所 有 这 些 工 作 都 是 在 由 根 类 定义 之 clone() 方 法 
的 内 部 代码 中 进行 的 〈 根 类 并 不 知道 要 从 目 己 这 里 继承 出 去 什么 ) ， 所 
以 大 家 或 许 已 经 猜 到 ， 这 个 过 程 需要 用 RTTI 判 断 欲 到 隆 的 对 象 的 实际 
大 小 。 采 取 这 种 方式 ，clone() 方 法 便 可 建立 起 正确 数量 的 存储 空间 ， 并 
对 那个 类 型 进行 正确 的 按 位 复制 。 


不 管 我 们 要 做 什么 ， 殉 隆 过 程 的 第 一 个 部 分 通常 都 应 该 是 调用 

super.cloneO0。 通 过 进行 一 次 准确 的 复制 ， 这 样 做 可 为 后 续 的 死 隆 进程 

人 
‘J it Pa o 

为 确切 了 解 其 他 操作 是 什么 ， 首 先 要 正确 理解 Object.clone() 为 我 们 带 来 
本 什么。 特别 地 ， 它 会 自动 克隆 所 有 人 句柄 指 问 的 目标 吗 ? 下 面 这 个 例子 

可 完成 这 种 形式 的 检测 : 


//: Snake.java 














// Tests cloning to see if destination of 
// handles are also cloned. 


public class Snake implements Cloneable { 


private Snake next; 
private char c; 
// Value of i == number of segments 
Snake(int i, char x) { 
Cc = xX} 
if(--i > 0) 
next = new Snake(i, (char)(x + 1)); 
} 
void increment() { 
C++; 
if(next != null) 
next.increment(); 
} 
public String toString() { 
String s = ":" + C; 
if(next != null) 
s += next.toString(); 
return s; 
} 
public Object clone() { 
Object o = null; 
try { 


o = super.clone(); 


} catch (CloneNotSupportedException e) {} 
return oO; 

} 

public static void main(String[] args) { 
Snake s = new Snake(5, 'a'); 
System.out.println("s = " + s); 
Snake s2 = (Snake)s.clone(); 
System.out.printiln("s2 = " + s2); 
s.increment(); 
System. out.printin( 

"after s.increment, s2 = " + s2); 
} 
} ///:~ 


一 条 Snake〈 蛇 ) 由 数 段 构成 ， 每 一 段 的 类 型 都 是 Snake。 所 以 ， 这 是 一 
个 一 段 段 链接 起 来 的 列表 。 所 有 上 段 都 是 以 循环 方式 创建 的 ， 每 做 好 一 
段 ， 都 会 使 第 一 个 构建 器 参数 的 值 递 减 ， 直 至 最 终 为 零 。 而 为 给 每 段 赋 
予 一 个 独一无二 的 标记 ， 第 二 个 参数 〈 一 个 Char) 的 值 在 每 次 循环 构建 
器 调用 时 都 会 递增 。 

increment() 方 法 的 作用 是 循环 递增 每 个 标记 ， 使 我 们 能 看 到 发 生 的 变 
化 ; 而 toString 则 循环 打印 出 每 个 标记 。 输 出 如 下 : 


S = :a:b:c:di:e 


s2 = :a:b:c:d:e 


after s.increment, s2 = :a:c:d:e:f 


这 意味 着 只 有 第 一 段 才 是 由 Object.clone0 复 制 的 ， 所 以 此 时 进行 的 是 一 
种 “ 浅 层 复制 >”。 若 希望 复制 整 条 蛇 一 一 即 进 行 “ 深 层 复 制 ” 一 一 必须 在 被 
覆盖 的 clone() 里 采取 附加 的 操作 。 


通常 可 在 从 一 个 能 克隆 的 类 里 调用 super.clone()， 以 确保 所 有 基础 类 行 
z) (包括 Object.clone()) 能 够 进行 。 随 着 是 为 对 象 内 每 个 句柄 都 明确 调 
用 一 个 clone(); 盏 则 那些 句柄 会 别名 变 成 原始 对 象 的 句柄 。 构 建 器 的 调 
用 也 大 致 相同 首先 构造 基础 类 ， 然 后 是 下 一 个 衍生 的 构建 器 .…… 以 
此 类 推 ， 直 到 位 于 最 深层 的 衍生 构建 器 。 区 别 在 于 clone() 并 不 是 个 构建 
器 ， 所 以 没有 办 法 实现 自动 克隆 。 为 了 克隆 ， 必 须 由 自己 明确 进行 。 


12.2.6 克隆 合成 对 象 


试图 深层 复制 合成 对 象 时 会 遇 到 一 个 问题 。 必 须 假 定 成 员 对 象 中 的 
clone() 方 法 也 能 依次 对 目 己 的 句柄 进行 深层 复制 ， 以 此 类 推 。 这 使 我 们 
的 操作 变 得 复杂 。 为 了 能 正常 实现 深层 复制 ， 必 须 对 所 有 类 中 的 代码 进 
行 控制 ， 或 者 至 少 全 面 掌握 深层 复制 中 需要 涉及 的 类 ， 确 保 它们 自己 的 
深层 复制 能 正确 进行 。 


DN ae 了 面 对 一 个 合成 对 象 进行 深层 复制 时 需要 做 哪些 事 
情 : 






































//: DeepCopy.java 


// Cloning a composed object 
Class DepthReading implements Cloneable { 
private double depth; 
public DepthReading(double depth) { 
this.depth = depth; 
} 
public Object clone() { 


Object o = null; 


try { 


o = super.clone(); 
} catch (CloneNotSupportedException e) { 
e.printStackTrace(); 


} 


return oO; 


} 
class TemperatureReading implements Cloneable { 
private long time; 
private double temperature; 
public TemperatureReading(double temperature) { 
time = System.currentTimeMillis(); 
this.temperature = temperature; 
} 
public Object clone() { 
Object o = null; 
try { 
o = super.clone(); 
} catch (CloneNotSupportedException e) { 
e.printStackTrace(); 


} 


return oO; 


} 


class OceanReading implements Cloneable { 
private DepthReading depth; 
private TemperatureReading temperature; 
public OceanReading(double tdata, double ddata){ 
temperature = new TemperatureReading(tdata); 
depth = new DepthReading(ddata); 
} 
public Object clone() { 
OceanReading 0 = null; 
try { 
o = (OceanReading)super.clone(); 
} catch (CloneNotSupportedException e) { 
e.printStackTrace(); 
} 
// Must clone handles: 
o.depth = (DepthReading)o.depth.clone(); 
o.temperature = 
(TemperatureReading)o.temperature.clone(); 


return o; // Upcasts back to Object 


public class DeepCopy { 
public static void main(String[] args) { 
OceanReading reading = 
new OceanReading(33.9, 100.5); 
// Now Clone it: 
OceanReading r = 
(OceanReading )reading.clone(); 
} 


和 As 


DepthReading 和 TemperatureReading 非 常 相似 ， 它 们 都 只 包含 了 基本 数 
据 类 型 。 所 以 clone0) 方 法 能 够 非常 简单 : 调用 super.clone() 并 返回 结果 即 
可 。 注 意 两 个 类 使 用 的 clone0 代 码 是 完全 一 致 的 。 


OceanReading 是 由 DepthReading 和 TemperatureReading 对 象 合 并 而 成 的 。 

为 了 对 其 进行 深层 复制 ，clone() 必 须 同时 元 隆 OceanReading 内 的 句柄 。 

为 达到 这 个 目标 ，super.clone0O 的 结果 必须 造型 成 一 个 OceanReading 对 象 
(以 便 访 问 depth 和 temperature 句 柄 ) 。 

12.2.7 用 Vector 进行 深层 复制 


下 面 让 我 们 复习 一 下 本 章 早 些 时候 提 出 的 Vector 例子 。 这 一 次 Int2 类 是 
可 以 克隆 的 ， 所 以 能 对 Vector 进行 深层 复制 : 


//: AddingClone.java 











// You must go through a few gyrations to 
// add cloning to your own class. 


import java.util.*; 


class Int2 implements Cloneable { 
private int i; 
public Int2(int ii) { i = ii; } 
public void increment() { i++; } 
public String toString() { 
return Integer.toString(i); 
} 
public Object clone() { 
Object 0 = null; 
try { 
o = super.clone(); 
} catch (CloneNotSupportedException e) { 
System.out.println("Int2 can't clone"); 


} 


return oO; 


} 


// Once it's cloneable, inheritance 
// doesn't remove cloneability: 
class Int3 extends Int2 { 
private int j; // Automatically duplicated 
public Int3(int i) { Super(1i); } 


} 


public class AddingClone { 
public static void main(String[] args) { 
Int2 x = new Int2(10); 
Int2 x2 = (Int2)x.clone(); 
x2.increment(); 
System.out.printin( 
SE eee EO, DS RE e 
// Anything inherited is also cloneable: 
Int3 x3 = new Int3(7); 
x3 = (Int3)x3.clone(); 
Vector v = new Vector(); 
for(int i = 0; i < 10; i++ ) 
v.addElement(new Int2(i)); 
System.out.printlin("v: " + v); 
Vector v2 = (Vector)v.clone(); 
// Now clone each element: 
for(int i = 0; i < v.size(); i++) 
v2.setElementAt ( 
((Int2)v2.elementAt(i)).clone(), 1); 
// Increment all v2's elements: 
for(Enumeration e = v2.elements(); 
e.hasMoreElements(); ) 


((Int2)e.nextElement()).increment(); 


// See if it changed v's elements: 
System.out.printlin("v: " + v); 
System.out.printin("v2: " + v2); 
} 
} ///:~ 





Int3 自 Int2 继 承 而 来 ， 并 添加 了 一 个 新 的 基本 类 型 成 员 int j。 大 家 也 许 认 
为 自己 需要 再 次 上 覆 善 cone(0)， 以 确保 j 得 到 复制 ， 但 实情 并 非 如 此 。 将 
Int2 的 clone0 当 作 Int3 的 cloneO 调 用 时 ， 它 会 调用 Object.clone0， 判 断 出 
当前 操作 的 是 Int3， 并 复制 Int3 内 的 所 有 二 进 制 位 。 只 要 没有 新 增 需要 
克隆 的 句柄 ， 对 Object.clone() 的 一 个 调用 就 能 完成 所 有 必要 的 复制 一 一 
无 论 clone0 是 在 层次 结构 多 深 的 一 级 定义 的 。 

至 此 ， 大 家 可 以 总 结 出 对 Vector 进行 深层 复制 的 先决 条 件 : 在 克隆 了 
Vector 后 ， 必 须 在 其 中 遍历 ， 并 克隆 由 Vector 指向 的 每 个 对 象 。 为 了 对 
Hashtable (HIK) 进行 深层 复制 ， 也 必须 采取 类 似 的 处 理 。 


这 个 例子 剩余 的 部 分 显示 出 克隆 已 实际 进行 一 一 证 据 束 是 在 元 隆 了 对 象 
以 后 ， 可 以 自由 改变 它 ， 而 原来 那个 对 象 不 受 任何 影响 。 


12.2.8 通过 序列 化 进行 深层 复制 
若 研 究 一 下 第 10 章 介绍 的 那个 Java ”1.1 对象 序列 化 示例 ， 可 能 发 现 若 在 
一 个 对 象 序列 化 以 后 再 撤消 对 它 的 序列 化 ， 或 者 说 进行 装配 ， 那 么 实际 
经 历 的 正 是 一 个 “克隆 ”的 过 程 。 


那么 为 什么 不 用 序列 化 进行 深层 复制 呢 ? 下 面 这 个 例子 通过 计算 执行 时 
间 对 比 了 这 两 种 方法 : 


//: Compete.java 

















import java.io.*; 


Class Thing1i implements Serializable {} 


class Thing2 implements Serializable { 
Thingi o1 = new Thing1(); 
} 
class Thing3 implements Cloneable { 
public Object clone() { 
Object o = null; 
try { 
0 = super.clone(); 
} catch (CloneNotSupportedException e) { 
System.out.printin("Thing3 can't clone"); 


} 


return oO; 


} 


class Thing4 implements Cloneable { 
Thing3 03 = new Thing3(); 
public Object clone() { 
Thing4 o = null; 
try { 
o = (Thing4)super.clone(); 
} catch (CloneNotSupportedException e) { 


System.out.printin("Thing4 can't clone"); 


// Clone the field, too: 
0.03 = (Thing3)o03.clone(); 


return oO; 


} 


public class Compete { 
static final int SIZE = 5000; 
public static void main(String[] args) { 
Thing2[] a = new Thing2[SIZE]; 
for(int i = 0; i < a.length; i++) 
a[i] = new Thing2(); 
Thing4[] b = new Thing4[SIZE]; 
for(int i = 0; i < b.length; i++) 
b[i] = new Thing4(); 
try { 
long t1 = System.currentTimeMillis(),; 
ByteArrayOutputStream buf = 
new ByteArrayOutputStream(); 
ObjectOutputStream o = 
new ObjectOutputStream(buf); 
for(int i = 0; i < a.length; i++) 
o.writeObject(a[i]); 


// Now get copies: 


ObjectInputStream in = 

new ObjectInputStream( 

new ByteArrayInputStream( 
buf.toByteArray())); 

Thing2[] c = new Thing2[SIZE]; 
for(int i = 0; i < c.length; i++) 

c[i] = (Thing2)in.readObject(); 
long t2 = System.currentTimeMillis(); 
System.out.printin( 

"Duplication via serialization: " + 

(t2 - t1) + " Milliseconds"); 
// Now try cloning: 
t1 = System.currentTimeMillis(); 
Thing4[] d = new Thing4[SIZE]; 
for(int i = 0; i < d.length; i++) 

d[i] = (Thing4)b[i].clone(); 
t2 = System.currentTimeMillis(); 
System. out.printin( 

"Duplication via cloning: " + 

(t2 - t1) + " Milliseconds"); 
catch(Exception e) { 


e.printStackTrace(); 


} 
} ///:~ 


其 中 ，Thing2 和 Thing4 包 含 了 成 员 对 象 ， 所 以 需要 进行 一 些 深层 复制 。 
一 个 有 趣 的 地 方 是 尽管 Serializable 类 很 容易 设置 ， 但 在 复制 它们 时 却 要 
做 多 得 多 的 工作 。 克 隆 涉及 到 大 量 的 类 设置 工作 ， 但 实际 的 对 象 复制 是 
相当 简单 的 。 结 果 很 好 地 说 明了 一 切 。 下 面 是 几 次 运行 分 别 得 到 的 结 
R: 


的 确 


Duplication via serialization: 3400 Milliseconds 








Duplication via cloning: 110 Milliseconds 
Duplication via serialization: 3410 Milliseconds 
Duplication via cloning: 110 Milliseconds 
Duplication via serialization: 3520 Milliseconds 


Duplication via cloning: 110 Milliseconds 


除了 序列 化 和 克隆 之 间 巨 大 的 时 间 差 异 以 外 ， 我 们 也 注意 到 序列 化 技术 
的 运行 结果 并 不 稳定 ， 而 殉 隆 每 一 次 花费 的 时 间 都 是 相同 的 。 


12.2.9 使 克隆 具有 更 大 的 深度 

若 新 建 一 个 类 ， 它 的 基础 类 会 默认 为 Object， 并 默认 为 不 具备 克隆 能 力 
(就 象 在 下 一 节 会 看 到 的 那样 ) 。 只 要 不 明确 地 添加 克隆 能 力 ， 这 种 能 
力 便 不 会 自动 产生 。 但 我 们 可 以 在 任何 层 添加 它 ， 然 后 便 可 从 那个 层 开 
始 向 下 具有 克隆 能 力 。 如 下 所 示 : 


//: HorrorFlick.java 





// You can insert Cloneability at any 


// level of inheritance. 
import java.util.*; 
class Person {} 
class Hero extends Person {} 
class Scientist extends Person 
implements Cloneable { 
public Object clone() { 
try { 
return super.clone(); 
} catch (CloneNotSupportedException e) { 
// this should never happen: 
// It's Cloneable already! 


throw new InternalError(); 


} 


} 


Class MadScientist extends Scientist {} 
public class HorrorFlick { 
public static void main(String[] args) { 
Person p = new Person(); 
Hero h = new Hero(); 
Scientist s = new Scientist(); 


MadScientist m = new MadScientist(); 


// p = (Person)p.clone(); // Compile error 
// h = (Hero)h.clone(); // Compile error 

s = (Scientist)s.clone(); 

m = (MadScientist)m.clone(); 


} 
} ///:~ 


添加 克隆 能 力 之 前 ， 编 译 器 会 阻止 我 们 的 元 隆 尝 试 。 一 旦 在 Scientist 里 
添加 了 克隆 能 力 ， 那 么 Scientist 以 及 它 的 所 有 “后 裔 ”都 可 以 克隆。 


12.2.10 为 什么 有 这 个 奇怪 的 设计 


之 所 以 感觉 这 个 方案 的 奇特 ， 因 为 它 事实 上 的 确 如 此 。 也 许 大 家 会 奇怪 

它 为 什么 要 象 这 样 运行 ， 而 该 方案 背后 的 真正 含义 是 什么 呢 ? 后 面 讲述 

的 是 一 个 未 获 证 实 的 故事 一 一 大 概 是 由 于 围绕 Java 的 许多 买卖 使 其 成 为 

恨 的 语言 一 一 但 确实 要 花 许 多 口舌 才能 讲 清楚 这 背后 发 生 的 
Ħ o 


最 初 ，Java 只 是 作为 一 种 用 于 控制 硬件 的 语言 而 设计 ， 与 因特网 并 没有 
丝 坚 联系 。 象 这 样 一 类 面向 大 众 的 语言 一 样 ， 其 意义 在 于 程序 员 可 以 对 
任意 一 个 对 象 进 行 殉 隆 。 这 样 一 来 ，clone() 束 放置 在 根 类 Object 里 面 ， 
但 因为 它 是 一 种 公用 方式 ， 因 而 我 们 通常 能 够 对 任意 一 个 对 象 进行 殉 
隆 。 看 来 这 是 最 灵活 的 方式 了 ， 上 毕竟 它 不 会 种 来 任何 害处 。 


正当 Java 看 起 来 象 一 种 终 级 因特网 程序 设计 语言 的 时 候 ， 情 况 却 发 生 了 
变化 。 突 然 地 ， 人 们 提出 了 安全 问题 ， 而 且 理 所 当然 ， 这 些 问 题 与 使 用 
对 象 有 关 ， 我 们 不 愿望 任何 人 克隆 自己 的 保密 对 象 。 所 以 我 们 最 后 看 到 
的 是 为 原来 那个 简单 、 直 观 的 方案 添加 的 大 量 补丁 : cloneO 在 Object 里 
被 设置 成 “protected”。 必 须 将 其 履 盖 ， 并 使 用 *implement Cloneable”, |E] 
时 解决 违例 的 问题 。 


只 有 在 准备 调用 Object 的 clone0 方 法 时 ， 才 没有 必要 使 用 Cloneable 接 
口 ， 因 为 那个 方法 会 在 运行 期 间 得 到 检查 ， 以 确保 我 们 的 类 实现 了 
Cloneable。 但 为 了 保持 连贯 性 《而 且 由 于 Cloneable 无 论 如 何 都 是 空 





























的 ) ， 最 好 还 是 由 自己 实现 Cloneable。 


12.3 克隆 的 控制 


为 消除 区 隆 能 力 ， 大 家 也 许 认 为 只 需 将 clone() 方 法 简单 地 设 为 

private CAA) 即 可 ， 但 这 样 是 行 不 通 的 ， 因 为 不 能 采用 一 个 基础 类 方 
法 ， 并 使 其 在 衍生 类 中 更 “私有 ”。 上 所 以 事情 并 没有 这 么 简单 。 此 外 ， 我 
们 有 必要 控制 一 个 对 象 是 否 能 够 元 隆 。 对 于 我 们 设计 的 一 个 类 ， 实 际 有 
许多 种 方案 都 是 可 以 采取 的 : 


(1) 保持 中 立 ， 不 为 克隆 做 任何 事情 。 也 就 是 说 ， 尺 管 不 可 对 我 们 的 类 
克隆 ， 但 从 它 继 承 的 一 个 类 却 可 根据 实际 情况 决定 克隆 。 只 有 

fee eae 类 中 的 字段 进行 条 些 合理 的 操作 时 ， 才 可 以 作 这 方面 
I 决定 。 


(2) ”支持 clone()， 采 用 实现 Cloneable (可 克隆 ) 能 力 的 标准 操作 ， 并 履 
盖 clone0。 在 被 覆盖 的 clone0 中 ， 可 调用 super.clone0， 并 捕获 所 有 违例 
(这 样 可 使 one0 不 “ 搓 ? 出 任何 违例 ) 。 


(3) 有 条 件 地 文 持 殉 隆 。 知 类 容纳 了 其 他 对 象 的 句柄 ， 而 那些 对 象 也 许 
能 够 区 隆 〈 集 合 类 便 是 这 样 的 一 个 例子 ) ， 就 可 试 痢 克隆 拥有 对 方 句 柄 
的 所 有 对 象 ; 如 果 它 们 “ 撕 ”* 出 了 违例 ， 只 需 让 这 些 违例 通过 即 可 。 举 个 
例子 来 说 ， 假 设 有 一 个 特殊 的 Vector， 它 试图 克隆 自己 容纳 的 所 有 对 
象 。 编 写 这 样 的 一 个 Vector 时 ， 并 不 知道 客户 程序 员 会 把 什么 形式 的 对 
象 置 入 这 个 Vector 中 ， 所 以 并 不 知道 它们 是 否 真 的 能 够 元 隆 。 


(4) 不 实现 Cloneable0， 但 是 将 clone0 履 盖 成 protected， 使 任何 字段 都 具 
有 正确 的 复制 行为 。 这 样 一 来 ， 从 这 个 类 继承 的 所 有 东西 都 能 窗 盖 
clone()， 并 调用 super.clone() 来 产生 正确 的 复制 行为 。 注 意 在 我 们 实现 方 
案 里 ， 可 以 而 且 应 该 调用 super.clone() 即使 那个 方法 本 来 预期 的 是 
一 个 Cloneable 对 象 〈 人 否则 会 掷 出 一 个 违例 ) ， 因 为 没有 人 会 在 我 们 这 种 
类 型 的 对 象 上 直接 调用 它 。 它 只 有 通过 一 个 衍生 类 调用 ;， 对 那个 衍生 类 
来 说 ， 如 果 要 保证 它 正 常 工作 ， 需 实现 Cloneable。 


(5) 不 实现 Cloneable 来 试 着 防止 元 隆 ， 并 和 窗 瘟 clone()， 以 产生 一 个 违 
例 。 为 使 这 一 设想 顺利 实现 ， 只 有 令 从 它 衍 生出 来 的 任何 类 都 调用 重新 
定义 后 的 done() 里 的 suepr.clone()。 























(6) 将 类 设 为 final， 从 而 防止 克隆 。 若 clone0) 尚 未 被 我 们 的 任何 一 个 上 级 
At, KM-KREERSK. SORE, PARKER, 

并 * 掷 ?出 一 个 CloneNotSupportedException (克隆 不 支持 ) 违例 。 为 担保 
克隆 被 禁止 ， 将 类 设 为 final 是 唯一 的 办 法 。 除 此 以 外 ， 一 旦 涉及 保密 对 
象 或 者 过 到 想 对 创建 的 对 象 数 量 进行 控制 的 其 他 情况 ， 应 该 将 所 有 构建 
器 都 设 为 private， 并 提供 一 个 或 更 多 的 特殊 方法 来 创建 对 象 。 采 用 这 种 
方式 ， 这 些 方法 就 可 以 限制 创建 的 对 象 数 量 以 及 它们 的 创建 条 件 一 一 一 
种 特殊 情况 是 第 16 章 要 介绍 的 singleton (独子 ) 方案 。 


下 面 这 个 例子 总 结 了 克隆 的 各 种 实现 方法 ， 然 后 在 层次 结构 中 将 其 * 关 
Hj”: 

















//: CheckCloneable.java 


// Checking to see if a handle can be cloned 
// Can't clone this because it doesn't 
// override clone(): 
class Ordinary {} 
// Overrides clone, but doesn't implement 
// Cloneable: 
class WrongClone extends Ordinary { 
public Object clone() 
throws CloneNotSupportedException { 


return super.clone(); // Throws exception 


} 


// Does all the right things for cloning: 


class IsCloneable extends Ordinary 


implements Cloneable { 
public Object clone() 
throws CloneNotSupportedException { 


return super.clone(); 


} 


// Turn off cloning by throwing the exception: 
class NoMore extends IsCloneable { 
public Object clone() 
throws CloneNotSupportedException { 


throw new CloneNotSupportedException(); 


} 


class TryMore extends NoMore { 
public Object clone() 
throws CloneNotSupportedException { 
// Calls NoMore.clone(), throws exception: 


return super.clone(); 


} 
} 


class BackOn extends NoMore { 
private BackOn duplicate(BackOn b) { 


// Somehow make a copy of b 


// and return that copy. This is a dummy 
// copy, just to make the point: 
return new BackOn(); 

} 

public Object clone() { 

// Doesn't call NoMore.clone(): 

return duplicate(this); 

} 
} 


// Can't inherit from this, so can't override 
// the clone method like in BackOn: 
final class ReallyNoMore extends NoMore {} 
public class CheckCloneable { 
static Ordinary tryToClone(Ordinary ord) { 
String id = ord.getClass().getName(); 
Ordinary x = null; 
if(ord instanceof Cloneable) { 
try { 
System.out.printin("Attempting " + id); 
x = (Ordinary) ((IsCloneable)ord).clone(); 
System.out.printin("Cloned " + id); 
} catch(CloneNotSupportedException e) { 


System.out.printin( 


"Could not clone " + id); 


} 


return x; 
} 
public static void main(String[] args) { 
// Upcasting: 
Ordinary[] ord = { 
new IsCloneable(), 
new WrongClone(), 
new NoMore(), 
new TryMore(), 
new BackOn(), 
new ReallyNoMore(), 
}; 
Ordinary x = new Ordinary(); 
// This won't compile, since clone() is 
// protected in Object: 
//\ x = (Ordinary)x.clone(); 
// tryToClone() checks first to see if 
// a class implements Cloneable: 
for(int i = 0; i < ord.length; i++) 


tryToClone(ord[i]); 


} 
} ///:~ 





第 一 个 类 Ordinary 代 表 着 大 家 在 本 书 各 处 最 肖 见 到 的 类 ; ANSEF SEE, 
但 在 它 正式 应 用 以 后 ， 却 也 不 禁止 对 其 元 隆 。 但 假如 有 一 个 指向 
Ordinary 对 象 的 句 顶 ， 而 且 那 个 对 象 可 能 是 从 一 个 更 深 的 衍生 类 上 济 造 
型 来 的 ， 便 不 能 判断 它 到 底 能 不 能 元 隆 。 


WrongClone 类 揭示 了 实现 殉 隆 的 一 种 不 正确 途径 。 它 确实 罗 关 了 
Object.clone()， 并 将 那个 方法 设 为 public， 但 却 没有 实现 Cloneable。 所 
以 一 旦 发 出 对 super.clone0 的 调用 《〈 由 于 对 Object.clone0 的 一 个 调用 造成 
的 ) ， 便 会 无 情 地 掷 出 CloneNotSupportedException 违 例 。 


在 IsCloneable 中 ， 大 家 看 到 的 才 是 进行 克隆 的 各 种 正确 行动 : FC ae 
clone0， 并 实现 了 Cloneable。 但 是 ， 这 个 clone() 方 法 以 及 本 例 的 另外 几 
个 方法 并 不 捕获 CloneNotSupportedException 违 例 ， 而 是 任 由 它 通 过 ， 并 
传递 给 调用 者 。 随 后 ， 调 用 者 必须 用 一 个 try-catch 代 码 块 把 它 包 围 起 
来 。 在 我 们 自己 的 clone0) 方 法 中 ， 通 常 需要 在 clone() 内 部 捕获 
CloneNotSupportedException 违 例 ， 而 不 是 任 由 它 通 过 。 正 如 大 家 以 后 会 
理解 的 那样 ， 对 这 个 例子 来 襄 ， 让 它 通 过 是 最 正确 的 做 法 。 


类 NoMore 试 图 按照 Java 设 计 者 打算 的 那样 “关闭 ”元 隆 : 在 衍生 类 clone0) 
中 ， 我 们 掷 出 CloneNotSupportedException 违 例 。TryMore 类 中 的 cloneg) 
方法 正确 地 调用 super.clone(0)， 并 解析 成 NoMore.cloneO0， 后 者 据 出 一 个 
违例 并 禁止 元 隆 。 


但 在 已 被 宪 盖 的 done() 方 法 中 ， 假 大 程序 员 不 亲 守 调用 super.clone() 

的 “正确 ”方法 ， 又 会 出 现 什么 情况 呢 ? 在 BackOn 中 ， 大 家 可 看 到 实际 会 
发 生 什 么 。 这 个 类 用 一 个 独立 的 方法 duplicate() 制 作 当 前 对 象 的 一 个 副 
本 ， 并 在 clone() 内 部 调用 这 个 方法 ， 而 不 是 调用 super.clone()。 违 例 永远 
不 会 产生 ， 而 且 新 类 是 可 以 克隆 的 。 因 此 ， 我 们 不 能 依赖 < 掷 ?出 一 个 违 
例 的 方法 来 防止 产生 一 个 可 克隆 的 类 。 唯 一 安全 的 方法 在 ReallyNoMore 
中 得 到 了 演示 ， 它 设 为 final， 所 以 不 可 继承 。 这 意味 着 假如 clone() 在 
final 类 中 掷 出 了 一 个 违例 ， 便 不 能 通过 继承 来 进行 修改 ， 并 可 有 效 地 人 禁 
止 克 隆 ( 不 能 从 一 个 拥有 任意 继承 级 数 的 类 中 明确 调用 Object.clone(); 
只 能 调用 super.clone()， 它 只 可 访问 直接 基础 类 ) 。 因 此 ， 只 要 制作 一 














些 涉及 安全 问题 的 对 象 ， 就 最 好 把 那些 类 设 为 final。 


在 类 CheckCloneable 中 ， 我 们 看 到 的 第 一 个 类 是 tyToClone0， 它 能 接纳 
任何 Ordinary 对 象 ， 并 用 instanceof 检 和 碍 它 是 售 能 够 克隆 。 知 答案 是 肯定 
的 ， 就 将 对 象 造型 成 为 一 个 IsCloneable， 调 用 clone()， 并 将 结果 造型 回 
Ordinary， 最 后 捕获 有 可 能 产生 的 任何 违例 。 请 注意 用 运行 期 类 型 鉴定 
( 见 第 11 章 〉 打 印 出 类 名 ,使 自己 看 到 发 生 的 一 切 情况 。 


在 main() 中 ， 我 们 创建 了 不 同类 型 的 Ordinary 对 象 ， 并 在 数组 定义 中 上 

溯 造 型 成 为 Ordinary。 在 这 之 后 的 头 两 行 代码 创建 了 一 个 纯粹 的 

Ordinary 对 象 ， 并 试图 对 其 克隆 。 然 而 ， 这 些 代 码 不 会 得 到 编译 ， 因 为 

clone0O 是 Object 中 的 一 个 protected〈 受 到 保护 的 ) 方法 。 代 码 剩余 的 部 

并 试 着 克隆 每 个 对 象 ， 分 别 报告 它们 的 成 功 或 失败 。 输 
IF: 


Attempting IsCloneable 





Cloned IsCloneable 
Attempting NoMore 

Could not clone NoMore 
Attempting TryMore 
Could not clone TryMore 
Attempting BackOn 
Cloned BackOn 
Attempting ReallyNoMore 


Could not clone ReallyNoMore 


总 之 ， 如 果 和 希望 一 个 类 能 够 元 隆 ， 那 么 : 


(1) 实现 Cloneable 接 口 


(2) 履 盖 clone0) 


(3) 在 自己 的 doneO 中 调用 super.clone() 
(4) 在 自己 的 clone0 中 捕获 违例 

这 一 系列 步骤 能 达到 最 理想 的 效果 。 
12.3.1 副本 构建 器 
克隆 看 起 来 要 求 进行 非常 复杂 的 设置 ， 似 乎 还 该 有 男 一 种 替代 方案 。 一 
个 办 法 是 制作 特殊 的 构建 器 ， 令 其 负责 复制 一 个 对 象 。 在 C++ 中 ， 这 叫 


作 * 副 本 构建 器 >。 刚 开始 的 时 候 ， 这 好 象 是 一 种 非常 显然 的 解决 方案 
(如 果 你 是 C++ 程 厅 员 ， 这 个 方法 就 更 显 杀 切 )。 下 面 是 一 个 实际 的 例 


Ta 








//: CopyConstructor.java 


// A constructor for copying an object 


// of the same type, as an attempt to create 


// a local copy. 


class FruitQualities { 


private 
private 
private 
private 
private 


// etc. 


int 


int 


int 


int 


int 


weight; 
color; 
firmness; 
ripeness; 


smell; 


FruitQualities() { // Default constructor 


// do something meaningful... 


} 


// Other constructors: 
A eae te 
// Copy constructor: 
FruitQualities(FruitQualities f) { 
weight = f.weight; 
color = f.color; 
firmness = f.firmness; 
ripeness = f.ripeness; 
smell = f.smell; 


// etc. 


} 


class Seed { 
// Members... 
Seed() { /* Default constructor */ } 
Seed(Seed s) { /* Copy constructor */ } 
} 
class Fruit { 
private FruitQualities fq; 
private int seeds; 
private Seed[] s; 


Fruit(FruitQualities q, int seedCount ) 


fq = q; 
seeds = seedCount; 
s = new Seed[seeds]; 
for(int i = 0; i < seeds; i++) 
s[i] = new Seed(); 
} 
// Other constructors: 
IT pe 
// Copy constructor: 
Fruit(Fruit f) { 
fq = new FruitQualities(f.fq); 
seeds = f.seeds; 
// Call all Seed copy-constructors: 
for(int i = 0; i < seeds; i++) 
s[i] = new Seed(f.s[i]); 
// Other copy-construction activities... 

} 

// To allow derived constructors (or other 
// methods) to put in different qualities: 
protected void addQualities(FruitQualities q) { 

fq = q; 

} 


protected FruitQualities getQualities() { 


return fq; 


J 


class Tomato extends Fruit { 
Tomato() { 
super (new FruitQualities(), 100); 
小 
Tomato(Tomato t) { // Copy-constructor 
super(t); // Upcast for base copy-constructor 


// Other copy-construction activities... 


} 


class ZebraQualities extends FruitQualities { 
private int stripedness; 
ZebraQualities() { // Default constructor 
// do something meaningful... 
} 
ZebraQualities(ZebraQualities z) { 
super(z); 


stripedness = z.stripedness; 


} 


class GreenZebra extends Tomato { 


GreenZebra() { 
addQualities(new ZebraQualities()); 
} 
GreenZebra(GreenZebra g) { 
super(g); // Calls Tomato(Tomato) 
// Restore the right qualities: 
addQualities(new ZebraQualities()); 
} 
void evaluate() { 
ZebraQualities zq = 
(ZebraQualities)getQualities(); 
// Do something with the qualities 


Ss re 


} 


public class CopyConstructor { 
public static void ripen(Tomato t) { 
// Use the "copy constructor": 
t = new Tomato(t); 
System.out.printin("In ripen, t is a" + 
t.getClass().getName()); 


} 


public static void slice(Fruit f) { 


f = new Fruit(f); // Hmmm... will this work? 
System.out.printin("In slice, f isa" + 
f.getClass().getName()); 

} 

public static void main(String[] args) { 
Tomato tomato = new Tomato(); 
ripen(tomato); // OK 
slice(tomato); // OOPS! 
GreenZebra g = new GreenZebra(); 
ripen(g); // OOPS! 
slice(g); // OOPS! 
g.evaluate(); 

} 

-XA 





这 个 例子 第 一 眼看 上 去 显得 有 点 奇怪 。 不 同 水 果 的 质量 肯定 有 所 区 别 ， 
但 为 什么 只 ! 是 把 代表 那些 质量 的 数据 成 员 直接 置 入 Fruit 《水 果 ) 类 类 ? 有 
两 方面 可 能 的 原因 。 第 一 个 是 我 们 可 能 想 简便 地 插入 或 修改 质量 。 注意 
Fruit 有 一 个 protected (受到 保护 的 ) addQualities() 方 法 ， 它 允许 衍生 类 
来 进行 这 些 插入 或 修改 操作 “〈 大 家 或 许 会 认为 最 合乎 逻辑 的 做 法 是 在 
Fruit 中 使 用 一 个 protected 构 建 嚣 ， 用 它 获取 FruitQualities 参 数 ， 但 构建 
器 不 能 继承 ， 所 以 不 可 在 第 二 级 或 级 数 更 深 的 类 中 使 用 它 ) 。 通 过 将 水 
果 的 质量 置 入 一 个 独立 的 类 ee eee 其 中 包括 可 以 在 
特定 Fruit 对 象 的 存在 期 间 中 途 余 更 改 质 量 


之 所 以 将 FruitQualities 设 为 一 个 独立 的 对 象 ， 另 一 个 原因 是 考虑 到 我 们 
有 时 希望 添加 新 的 质量 ， 或 者 通过 继承 与 多 形 性 改变 行为 。 注 意 对 
GreenZebra 来 说 (这 实际 是 西红柿 的 一 类 我 已 栽种 成 功 ， 它 们 简直 














令 人 难以 置信 ) ， 构 建 器 会 调用 addQualities()， 并 为 其 传递 一 个 
ZebraQualities 对 象 。 该 对 象 是 从 EruitQualities 衍 生出 来 的 ， 所 以 能 与 基 
础 类 中 的 FruitQualities 句 柄 联系 在 一 起 。 当 然 ， 一 旦 日 GreenZebra 使 用 
FruitQualities， 束 必须 将 其 下 浏 造型 成 为 正确 的 类 型 ( 束 象 evaluate() 中 
展示 的 那样 )， 但 它 肯 定 知 道 类 型 是 ZebraQualities。 


大 家 也 看 到 有 一 个 Seed (FHF) ZA, Fruit (大 家 都 知道 ， 水 果 含 有 自己 
的 种 子 ) 包含 了 一 个 Seed 数 组 。 


最 后 ， 注 意 每 个 类 都 有 一 个 副本 构建 器 ， 而 且 每 个 副本 构建 器 都 必须 关 
心 为 基础 类 和 成 员 对 象 调用 副本 构建 器 的 问题 ， 从 而 获得 “深层 复制 ”的 
效果 。 对 副本 构建 器 的 测试 是 在 CopyConstructor 类 内 进行 的 。 方 法 
ay See 并 对 其 执行 副本 构建 工作 ， 以 便 复 制 
XZ: 














t = new Tomato(t); 

而 sliceO 需 要 获取 一 个 更 常规 的 Fruit 对 象 ， 而 且 对 它 进 行 复制 : 

f = new Fruit(f); 

它们 都 在 main0 中 伴随 不 同 种 类 的 Fruit 进 行 测试 。 下 面 是 输出 结果 : 


In ripen，t is a Tomato 


In slice, f is a Fruit 
In ripen, t is a Tomato 


In slice, f is a Fruit 


从 中 可 以 看 出 一 个 问题 。 在 sliceO0 内 部 对 Tomato 进 行 了 副本 构建 工作 以 
后 ， 结 果 便 不 再 是 一 个 Tomato 对 象 ， 而 只 是 一 个 Fruit。 它 已 丢失 了 作为 
一 个 Tomato 〈 西 红 柿 ) 的 所 有 特征 。 此 外 ， 如 果 采 用 一 个 GreenZebra， 

ripen() 和 和 sliceO 会 把 它 分 别 转换 成 一 个 Tomato 和 一 个 Fruit。 所 以 非常 不 

Se: eee Java 中 的 副本 构建 器 便 不 是 特别 
适合 我 们 。 


1. 为 什么 在 C++ 的 作用 比 在 Java 中 大 ? 


副本 构建 器 是 C++ 的 一 个 基本 构成 部 分 ， 因 为 它 能 目 动 产生 对 象 的 一 个 
本 地 副本 。 但 前 面 的 例子 确实 证 明了 它 不 适合 在 Java 中 使 用 ， 为 什么 
呢 ? 在 Java 中 ， 我 们 操控 的 一 切 东西 都 是 句柄 ， 而 在 C++ 中 ， 却 可 以 使 
用 类 似 于 句柄 的 东西 ， 也 能 直接 传递 对 象 。 这 时 便 要 用 到 C++ 的 副本 构 
idk: 只 要 想 获得 一 个 对 象 ， 并 按 值 传递 它 ， 就 可 以 复制 对 象 。 所 以 它 
T 但 应 注意 这 套 机 制 在 Java 里 是 很 不 通 的 ， 所 以 
NEE FE 








12.4 只 读 类 


尽管 在 一 些 特 定 的 场合 ， 由 cloneO 产 生 的 本 地 副本 能 够 获得 我 们 希望 的 
结果 ， 但 程序 员 ( 方 法 的 作者 〉 不 得 不 亲自 禁止 别名 处 理 的 副作用 。 假 
如 想 制作 一 个 库 ， 令 其 具有 常规 用 途 ， 但 却 不 能 担保 它 肯 定 能 在 正确 的 
类 中 得 以 殉 隆 ， 这 时 又 该 怎么 办 呢 ? 更 有 可 能 的 一 种 情况 是 ， 假 如 我 们 
想 让 别名 发 挥 积极 的 作用 一 一 禁止 不 必要 的 对 象 复 制 一 一 但 却 不 希望 看 
到 由 此 造成 的 副作用 ， 那 么 又 该 如 何 处 理 呢 ? 


一 个 办 法 是 创建 < 不 变 对 象 ”， 令 其 从 属于 只 读 类 。 可 定义 一 个 特殊 的 
类 ， 使 其 中 没有 任何 方法 能 造成 对 象 内 部 状态 的 改变 。 在 这 样 的 一 个 类 
中 ， 别 名 处 理 是 没有 问题 的 。 因 为 我 们 只 能 读 取 内 部 状态 ， 所 以 当 多 处 
代码 都 读 取 相同 的 对 象 时 ， 不 会 出 现任 何 副作用 。 

作为 "不 变 对 象 "一 个 简单 例子 ，Java 的 标准 库 包含 了 “封装 

器 ”(wrapper) 类 ， 可 用 于 所 有 基本 数据 类 型 。 大 家 可 能 已 发 现 了 这 一 
点 ， 如 果 想 在 一 个 象 Vector (只 采用 Object 名 柄 ) 这 样 的 集合 里 保存 一 
个 in 数值， 可 以 将 这 个 int 封 装 到 标准 库 的 Integer 交 内部。 如 下 所 示 : 


//: ImmutableInteger.java 

















// The Integer class cannot be changed 
import java.util.*; 
public class ImmutableInteger { 
public static void main(String[] args) { 
Vector v = new Vector(); 
for(int i = 0; i < 10; i++) 
v.addElement(new Integer(i)); 
// But how do you change the int 


// inside the Integer? 


} 
于 LA 


Integer 类 《以 及 基本 的 “封装 器 ?类 ) 用 简单 的 形式 实现 了 “不 变性 ”， 它 
们 没有 提供 可 以 修改 对 象 的 方法 。 


石 确 实 需要 一 个 容纳 了 基本 数据 类 型 的 对 象 ， 并 想 对 基本 数据 类 型 进行 
修改 ， 就 必须 杀 自 创建 它们 。 笠 运 的 是 ， 操 作 非 常 简单 : 


//: MutableInteger.java 








// A changeable wrapper class 
import java.util.*; 
class IntValue { 
int n; 
IntValue(int x) { n = x; } 
public String toString() { 


return Integer.toString(n); 


} 
public class MutableInteger { 
public static void main(String[] args) { 
Vector v = new Vector(); 
for(int i = 0; i < 10; i++) 
v.addElement(new IntValue(i)); 


System.out.println(v); 


for(int i = 0; i < v.size(); i++) 
((IntValue)v.elementAt(i)).n++; 
System.out.println(v); 


} 
} ///:~ 


注意 n 在 这 里 简化 了 我 们 的 编码 。 


耕 默 认 的 初始 化 为 零 已 经 足够 ( 便 不 需要 构建 器 ) ， 而 且 不 用 考虑 把 它 
打印 出 来 〈 便 不 需要 toString) ， 那 么 IntValue 甚 至 还 能 更 加 简单 。 如 下 
ARAN: 


class IntValue { int n; } 


WotR AH DOK, Foe Bok Tika, mea Ae, (AAS +E Vector 
的 问题 ， 不 是 IntValue 的 错 。 


12.4.1 创建 只 读 类 
完全 可 以 创建 自己 的 只 读 类 ， 下 面 是 个 简单 的 例子 : 


//: Immutable1.java 


// Objects that cannot be modified 
// are immune to aliasing. 
public class Immutable1 { 

private int data; 

public Immutablei(int initVal) { 


data = initVal; 


public int read() { return data; } 
public boolean nonzero() { return data != 0; } 
public Immutable1 quadruple() { 
return new Immutablei(data * 4); 
} 
static void f(Immutable1 i1) { 
Immutable1 quad = i1.quadruple(); 
System.out.println("ii = " + i1.read()); 
System.out.println("quad = " + quad.read()); 
} 
public static void main(String[] args) { 


Immutable1 x = new Immutable1(47); 


System.out.printin("x = " + x.read()); 
f(x); 
System.out.println("x = " + x.read()); 
Í 
} ///:~ 


所 有 数据 都 设 为 private， 可 以 看 到 没有 任何 public 方 法 对 数据 作出 修 
改 。 事 实 上 ， 确 实 需要 修改 一 个 对 象 的 方法 是 quadruple()， 但 它 的 作用 
是 新 建 一 个 Immutable1 对 象 ， 初 始 对 象 则 是 原封 未 动 的 。 


方法 们 需要 取得 一 个 Immutable1 对 象 ， 并 对 其 采取 不 同 的 操作 ， 而 
main0 的 输出 显示 出 没有 对 x 作 任何 修改 。 因 此 ，x 对 象 可 别名 处 理 许多 
ee 因为 根据 Immutablel 类 的 设计 ， 它 能 保证 对 象 
不 被 改动 。 





12.4.2 “一 成 不 变 ” 的 浆 端 

从 表面 看 ， 不 变 类 的 建立 似乎 是 一 个 好 方案 。 但 是 ， 一 旦 真 的 需要 那 种 
新 类 型 的 一 个 修改 的 对 象 ， 就 必须 辛苦 地 进行 新 对 象 的 创建 工作 ， 同 时 
还 有 可 能 涉及 更 频繁 的 垃圾 收集 。 对 有 些 类 来 说 ， 这 个 问题 并 不 是 很 
大 。 但 对 其 他 类 来 说 (比如 String 类 ) ， 这 一 方案 的 代价 显得 太 高 了 。 
为 解决 这 个 问题 ， 我 们 可 以 创建 一 个 “同志 ”类 ， 并 使 其 能 够 修改 。 以 后 
只 要 涉及 大 量 的 修改 工作 ， 就 可 换 为 使 用 能 修改 的 同志 类 。 完 事 以 后 ， 
再 切换 回 不 可 变 的 类 。 

因此 ， 上 例 可 改 成 下 面 这 个 样子 : 


//: Immutable2.java 





// A companion class for making changes 
// to immutable objects. 
class Mutable { 
private int data; 
public Mutable(int initVal) { 
data = initVal; 
} 
public Mutable add(int x) { 
data += x; 
return this; 
} 
public Mutable multiply(int x) { 
data *= x; 


return this; 


} 


public Immutable2 makeImmutable2() { 


return new Immutable2(data); 


} 


public class Immutable2 { 
private int data; 
public Immutable2(int initVal) { 
data = initVal; 
} 
public int read() { return data; } 
public boolean nonzero() { return data != 0; } 
public Immutable2 add(int x) { 
return new Immutable2(data + x); 
} 
public Immutable2 multiply(int x) { 
return new Immutable2(data * x); 
} 
public Mutable makeMutable() { 
return new Mutable(data); 
} 
public static Immutable2 modifyi(Immutable2 y){ 


Immutable2 val = y.add(12); 


val = val.multiply(3); 


val val.add(11); 
val = val.multiply(2); 
return val; 
} 
// This produces the same result: 
public static Immutable2 modify2(Immutable2 y){ 
Mutable m = y.makeMutable(); 
m.add(12).multiply(3).add(11).multiply(2); 
return m.makeImmutable2(); 
} 
public static void main(String[] args) { 
Immutable2 12 = new Immutable2(47); 
Immutable2 ri = modify1(i2); 


Immutable2 r2 = modify2(1i2); 


System.out.println("i2 = " + i2.read()); 
System.out.printin("ri = " + ri.read()); 
System.out.printin("r2 = " + r2.read()); 
} 
LIL 


和 往常 一 样 ，Immutable2 包 含 的 方法 保留 了 对 象 不 可 变 的 特征 ， 只 要 小 
及 修改 ， 束 创建 新 的 对 象 。 完 成 这 些 操作 的 是 addO) 和 multiply0 方 法 。 同 
志 类 叫 作 Mutable， 它 也 含有 addO 和 multiply0 方 法 。 但 这 些 方 法 能 够 修 


改 Mutable 对 象 ， 而 不 是 新 建 一 个 。 除 此 以 外 ，Mnutable 的 一 个 方法 可 用 
它 的 数据 产生 一 个 Immutable2 对 象 ， 反 之 亦 然 。 


两 个 静态 方法 modify10 和 modify20 揭 示 出 获得 同样 结果 的 两 种 不 同方 
法 。 在 modify10 中 ， 所 有 工作 都 是 在 Immutable2 类 中 完成 的 ， 我 们 可 看 
到 在 进程 中 创建 了 四 个 新 的 Inmutable2 对 象 〈 而 且 每 次 重新 分 配 了 val， 
前 一 个 对 象 就 成 为 垃圾 )。 

在 方法 modify20 中 ， 可 看 到 它 的 第 一 个 行动 是 获取 Immutable2 y， 然 后 
从 中 生成 一 个 Mutable 《类似 于 前 面 对 clone0 的 调用 ， 但 这 一 次 创建 了 一 
个 不 同类 型 的 对 象 )。 随 后 ， 用 Mutable 对 象 进行 大 量 修改 操作 ， 同 时 
用 不 着 新 建 许多 对 象 。 最 后 ， 它 切换 回 Immutable2。 在 这 里 ， 我 们 只 创 
建 了 两 个 新 对 象 (Mutable 和 Immutable2 的 结果 ) ， 而 不 是 四 个 。 

这 一 方法 特别 适合 在 下 述 场合 应 用 : 

(1) 需要 不 可 变 的 对 象 ， 而 且 

(2) 经 常 需要 进行 大 量 修 改 ， 或 者 

(3) 创建 新 的 不 变 对 象 代 价 太 高 

12.4.3 PEF E 

请 观察 下 述 代码 : 


//: Stringer.java 














public class Stringer { 
static String upcase(String s) { 
return s.toUpperCase(); 
} 
public static void main(String[] args) { 


String q = new String("howdy"); 


System.out.printin(q); // howdy 
String qq = upcase(q); 
System.out.println(qq); // HOWDY 
System.out.printin(q); // howdy 
} 
} ///:~ 





qd 传递 进入 upcaseO0 时 ， 它 实际 是 q 的 句柄 的 一 个 副本 。 该 句柄 连接 的 对 
Se Re eae eet ey ge 
得 到 复制 。 


在 观察 对 upcase0 的 定义 ， 会 发 现 传递 进入 的 句柄 有 一 个 名 字 s， 而 且 访 

名 字 只 有 在 upcase(0) 执 行 期 间 才 会 存在 。upcase0 完 成 后 ， 本 地 句柄 s 便 会 

消失 ， 而 upcase0 返 回 结果 一 一 还 是 原来 那个 字 串 ， 只 是 所 有 字符 都 变 

成 了 大 写 。 当 然 ， 它 返回 的 实际 是 结果 的 一 个 句柄 。 但 它 返 回 的 句柄 最 

i E E 
和 呢 ? 


1. 隐 式 常数 
若 使 用 下 述 语句 : 








String s = "asdf"; 
String x = Stringer.upcase(s); 


MARKA Bupcase() 77 AANE A OE BG? 我 们 通常 是 不 愿意 
的 ， 因 为 作为 提供 给 方法 的 一 种 信息 ， 自 变量 一 般 是 拿 给 代码 的 读者 看 
ar a 
编写 和 理解 。 


为 了 在 C++ 中 实现 这 一 保证 ， 需 要 一 个 特殊 关键 字 的 帮助 : const。 利 用 
这 个 关键 字 ， 程 序 员 可 以 保证 一 个 句柄 〈C++ 叫 “指针 ?或 者 “引用 ”) 不 
会 被 用 来 修改 原始 的 对 象 。 但 这 样 一 来 ，C++ 程 序 员 需要 用 心 记 住 在 所 














有 地 方 都 使 用 const。 这 显然 易 使 人 混 消 ， 也 不 容易 记 住 。 
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利用 前 面 提 到 的 技术 ，String 类 的 对 象 被 设计 成 “不 可 变 ”。 知 查阅 联机 
文档 中 关于 String 类 的 内 容 〈 本 章 稍 后 还 要 总 结 它 ) ， 就 会 发 现 类 中 能 
够 修改 String 的 每 个 方法 实际 都 创建 和 返回 了 一 个 轿 新 的 String 对 象 ， 新 
对 象 里 包含 了 修改 过 的 信息 原来 的 String 是 原封 未 动 的 。 因 此 ， 
Java 里 没有 与 C++ 的 const 对 应 的 特性 可 用 来 让 编译 器 支持 对 象 的 不 可 变 
能 力 。 知 想 获得 这 一 能 力 ， 可 以 上 自行 设置 ， 就 象 String 那 样 。 


由 于 String 对 象 是 不 可 变 的 ， 所 以 能 够 根据 情况 对 一 个 特定 的 String 进 行 
多 次 别名 处 理 。 因 为 它 是 只 读 的 ， 所 以 一 个 句柄 不 可 能 会 改变 一 些 会 影 
啊 其 他 句柄 的 东西 。 因 此 ， 只 读 对 象 可 以 很 好 地 解决 别名 问题 。 


通过 修改 产生 对 象 的 一 个 固 新 版 本 ， 似 乎 可 以 解决 修改 对 象 时 的 所 有 问 
题 ， 就 象 String 那 样 。 但 对 某 些 操作 来 讲 ， 这 种 方法 的 效率 并 不 高 。 一 
个 典型 的 例子 便 是 为 String 对 象 履 六 的 运算 符 “+”。“ 禾 新 ”意味 着 在 与 一 
个 特定 的 类 使 用 时 ， 它 的 含义 已 发 生 了 变化 (用 于 String 的 “+” 和 “+=” 是 
Java 中 能 被 覆盖 的 唯一 运算 符 ，Java 不 允许 程序 员 履 盖 其 他 任何 运算 符 
TEOD) 。 

O: C++ 人 允许 程序 员 随 意 履 盖 运 算 符 。 由 于 这 通常 是 一 个 复杂 的 过 程 
(参见 《Thinking in C++》，Prentice-Hall 于 1995 年 出 版 ) ， 所 以 Java 的 
设计 者 认定 它 是 一 种 “ 粳 糕 ?的 特性 ， 诀 定 不 在 Java 中 采用 。 但 具有 讽刺 
意味 的 是 ， 运 算 符 的 覆盖 在 Java 中 要 比 在 C++ 中 容易 得 多 。 

针对 String 对 象 使 用 时 ,，“+? 人 允许 我 们 将 不 同 的 字 串 连接 起 来 : 


String s = "abc" + foo + "def" + Integer.toString(47); 






































可 以 想象 出 它 “ 可 能 ?是 如 何 工 作 的 : 字 串 "abc" 可 以 有 一 个 方法 
append()， 它 新 建 了 一 个 字 串 ， 其 中 包含 "abc" 以 及 foo 的 内 容 这 个 新 字 
串 然后 再 创建 男 一 个 新 字 串 ， 在 其 中 添加 "def";， 以 此 类 推 。 


这 一 设想 是 行 得 通 的 ， 但 它 要 求 创 建 大 量 字 串 对 象 。 尽 管 最 终 的 目的 只 
古 获 得 包含 了 所 有 内 容 的 一 个 新 字 串 ， 但 中 间 却 要 用 到 大 量 字 串 对 象 ， 





而 且 要 不 断 地 进行 垃圾 收集 。 我 怀疑 Java 的 设计 者 是 否 先 试 过 种 方法 
(这 是 软件 开发 的 一 个 教训 一 一 除非 目 己 试 试 代 码 ， 并 让 茶 些 东西 运行 
起 来 ， 人 否则 不 可 能 真正 了 解 系统 ) 。 我 还 怀疑 他 们 是 否 早 就 发 现 这 样 做 
获得 的 性 能 是 不 能 接受 的 。 


解决 的 方法 是 象 前 面 介 绍 的 那样 制作 一 个 可 变 的 同志 类 。 对 字 串 来 说 ， 
这 个 同志 类 叫 作 StringBuffer， 编 译 器 可 以 自动 创建 一 个 StingBuffer， 以 
便 计 算 特 定 的 表达 式 ， 特 别 是 面向 String 对 象 应 用 履 盖 过 的 运算 符 + 和 += 
时 。 下 面 这 个 例子 可 以 解决 这 个 问题 : 


//: ImmutableStrings.java 




















// Demonstrating StringBuffer 
public class ImmutableStrings { 
public static void main(String[] args) { 
String foo = "foo"; 
String s = "abc" + foo + 
"def" + Integer.toString(47); 
System.out.println(s); 
// The "equivalent" using StringBuffer: 
StringBuffer sb = 
new StringBuffer("abc"); // Creates String! 
sb.append(foo); 
sb.append("def"); // Creates String! 
sb.append(Integer.toString(47)); 


System.out.println(sb); 


} ///:~ 


创建 字 串 s 时 ， 编 译 器 做 的 工作 大 致 等 价 于 后 面 使 用 sb 的 代码 一 一 创建 

一 个 StringBuffer， 并 用 appendO 将 新 字符 直接 加 入 StringBuffer 对 象 〈 而 

不 是 每 次 都 产生 新 对 象 ) 。 尽 管 这 样 做 更 有 效 ， 但 不 值得 每 次 都 创建 

象 "abc" 和 "def" 这 样 的 引号 字 囊 ， 编 译 吕 会 把 它们 都 转换 成 String 对 象 。 

4 Gece ee 但 会 产生 比 我 们 希望 的 多 得 多 
XY RR o 


12.4.4 String 和 StringBuffer 类 

这 里 总 结 一 下 同时 适用 于 String 和 StringBuffer 的 方法 ， 以 便 对 它们 相互 
间 的 沟通 方式 有 一 个 印象 。 这 些 表格 并 未 把 每 个 单独 的 方法 都 包括 进 
去 ， 而 是 包含 了 与 本 次 讨论 有 重要 关系 的 方法 。 那 些 已 被 履 盖 的 方法 用 
单独 一 行 总 结 。 

首先 总 结 String 类 的 各 种 方法 : 

方法 As, Bite 用 途 


构建 器 已 被 覆盖 : 默认 ，String，StringBuffer，char 数 组 ，byte 数 组 创 
建 String 对 象 


length) 无 String 中 的 字符 数量 














charAt() int Index 位 于 String 内 某 个 位 置 的 char 


getChars(), getBytes ”开始 复制 的 起 点 和 终点 ， 要 癌 其 中 复制 内 容 的 数 
组 ， 对 目标 数组 的 一 个 索引 将 char 或 byte 复 制 到 外 部 数组 内 部 


toCharArray() 无 产生 一 个 char[]， 其 中 包含 了 String 内 部 的 字符 


equals()，equalsIgnoreCase() 用 于 对 比 的 一 个 String 对 两 个 字 串 的 内 容 进 
行 等 价 性 检查 


compareTo() ”用 于 对 比 的 一 个 String ”结果 为 负 、 零 或 正 ， 有 具体 取决 于 
String 和 目 变 量 的 字典 顺序 。 注 意 大 写 和 小 写 不 是 相等 的 ! 





regionMatches() 这 个 String 以 及 其 他 String 的 位 置 偏 移 ， 以 及 要 比较 的 区 
域 长 度 。 窗 盖 加 入 了 “忽略 大 小 写 ” 的 特性 一 个 布尔 结果 ， 指 出 要 对 比 的 
区 域 是 否 相 同 


startsWith() 可 能 以 它 开头 的 String。 禾 新 在 自 变 量 里 加 入 了 偏 移 一 个 布 
尔 结果 ， 指 出 String 是 否 以 那个 自 变 量 开头 


endsWith() 可 能 是 这 个 String 后 级 的 一 个 String 一 个 布尔 结果 ， 指 出 自 变 
量 是 不 是 一 个 后 级 


indexOf(),lastIndexOf() 已 覆盖 : char，char 和 起 始 索引 ，String，String 和 
ERRI 辱 自 变量 未 在 这 个 String 里 找到 ， 则 返回 -1， 否 则 返回 自 变量 
开始 处 的 位 置 索引 。lastIndexOf0) 可 从 终点 开始 回溯 搜索 


substring0 C#m: HORS], HIRAI MARA] “返回 一 个 新 的 
String 对 象 ， 其 中 包含 了 指定 的 字符 子 集 


concat() 想 连 结 的 String 返回 一 个 新 String 对 象 ， 其 中 包含 了 原始 String 的 
字符 ， 并 在 后 面 加 上 由 自 变 量 提供 的 字符 


relpace() ”要 查找 的 老 字 符 ， 要 用 它 蔡 换 的 新 字符 ”返回 一 个 新 String 对 
象 ， 其 中 已 完成 了 蔡 换 工作 。 知 没有 找到 相符 的 搜索 项 ， 束 沿用 老 字 串 


toLowerCase(),toUpperCase() 无 返回 一 个 新 String 对 象 ， 其 中 所 有 字符 的 
大 小 写 形式 都 进行 了 统一 。 基 不必 修改 ， 则 沿用 老 字 串 


tim) 无 返回 一 个 新 的 String 对 象 ， 头 尾 空 日 均 己 删除 。 知 考 需 改动 ， 
Wye EB 


valueOf() Citi: object, char[], char[]#l (mt VL kit BL, boolean, 
char, int, long, float, double 返回 一 个 String， 其 中 包含 自 变 量 的 一 个 
字符 表现 形式 


nten) 无 ”为 每 个 独一无二 的 字符 顺序 都 产生 一 个 (而 且 只 有 一 个 ) 
String) 


可 以 看 到 ， 一 旦 有 必要 改变 原来 的 内 容 ， 每 个 String 方 法 都 小 心地 返回 
了 一 个 新 的 String 对 象 。 另 外 要 注意 的 一 个 问题 是 ， 知 内 容 不 需要 改 
变 ， 则 方法 只 返回 指向 原来 那个 String 的 一 个 句柄 。 这 样 做 可 以 节省 存 























储 空间 和 系统 开销 。 
下 面 列 出 有 关 StringBuffer (Ze) 类 的 方法 : 
方法 ARE, Ai 用 途 


构建 器 Cait: 默认 ， 要 创建 的 缓冲 区 长 度 ， 要 根据 它 创建 的 String 新 
2 —~SStringBufferXt Z% 











toString() 无 根据 这 个 StringBuffer 创 建 一 个 String 
length() 无 StringBuffer 中 的 字符 数量 
capacity() 无 返回 目前 分 配 的 空间 大 小 


ensureCapacity() 用 于 表示 和 希望 容量 的 一 个 整数 使 StringBuffer 容 纳 至 少 
希望 的 空间 大 小 


setLength() 用 于 指示 绥 冲 区 内 字 串 新 长 度 的 一 个 整数 缩短 或 扩充 前 一 个 
字符 串 。 如 果 是 扩充 ， 则 用 null 值 填充 空 际 


ae 表示 目标 元 素 所 在 位 置 的 一 个 整数 返回 位 于 缓冲 区 指定 位 置 处 
Jchar 


setCharAt() 代表 目标 元 素 位 置 的 一 个 整数 以 及 元 素 的 一 个 新 char 值 修改 
日 定 位 置 处 的 值 


getChars() 复制 的 起 点 和 终点 ， 要 在 其 中 复制 的 数组 以 及 目标 数组 的 一 
Ft 将 char 复 制 到 一 个 外 部 数组 。 和 String 不 同 ， 这 里 没有 getBytes0) 
可 供 使 用 


append() 已 覆盖 : Object，String，char[]， 特 定 偏 移 和 长 度 的 char[]， 
boolean, char, int, long, float, double 将 自 变 量 转换 成 一 个 字 串 ， 并 
将 其 奶 加 到 当前 缓冲 区 的 末尾 。 奉 有 必要 ， 同 时 增 大 绥 冲 区 的 长 度 


insert()” 己 禾 产 ， 第 一 个 自 变 量 代表 开始 插入 的 位 置 : Object, String, 
char[]，boolean，char，int，long，float，double 第 二 个 目 变 量 转换 成 一 
个 字 串 ， 并 插入 当前 缓冲 区 。 插 入 位 置 在 偏 移 区 域 的 起 点 处 。 奉 有 必 
要 ， 同 时 会 增 大 缓冲 区 的 长 度 











reverse() 无 反 转 绥 冲 内 的 字符 顺序 


最 第 用 的 一 个 方法 是 append0。 在 计算 包含 了 + 和 += 运 算 符 的 String 表 达 
式 时 ， 编 译 器 便 会 用 到 这 个 方法 。insert() 方 法 采用 类 似 的 形式 。 这 两 个 
方法 都 能 对 缓冲 区 进行 重要 的 操作 ， 不 需要 男 建新 对 象 。 


12.4.5 字 串 的 特殊 性 


现在 ， 大 家 已 知道 String 类 并 非 仅仅 是 Java 提 供 的 另 一 个 类 。String 里 含 
有 大 量 特殊 的 类 。 通 过 编译 器 和 特殊 的 履 盖 或 过 载运 算 符 + 和 +=， 可 将 
引号 字符 串 转 换 成 一 个 String。 在 本 章 中 ， 大 家 已 见识 了 剩 下 的 一 种 特 
殊 情 况 : 用 同志 StringBuffer 精 心 构造 的 “不 可 变 ” 能 力 ， 以 及 编译 器 中 出 
现 的 一 些 有 趣 现象 。 





12.5 总 结 


由 于 Java 中 的 所 有 东西 都 是 句柄 ， 而 且 由 于 每 个 对 象 都 是 在 内 存 堆 中 创 
建 的 一 一 只 有 不 再 需要 的 时 候 ， 才 会 当 作 垃圾 收集 挥 ， 所 以 对 象 的 操作 
方式 及 生 了 变化 ， 特 别 是 在 传递 和 返回 对 象 的 时 候 。 举 个 例子 来 说 ， 在 
C 和 C++ 中 ， 如 果 想 在 一 个 方法 里 初始 化 一 些 存储 空间 ， 可 能 需要 请 求 

用 户 将 那 片 存储 区 域 的 地 址 传递 进入 方法 。 否 则 就 必须 考虑 由 谁 负责 清 
除 那 片 区 域 。 因 此 ， 这 些 方法 的 接口 和 对 它们 的 理解 就 显得 要 复杂 一 

些 。 但 在 Java 中 ， 根 本 不 必 关 心 由 谁 负 责 清除 ， 也 不 必 关 心 在 需要 一 个 
对 象 的 时 候 它 是 否 仍然 存在 。 因 为 系统 会 为 我 们 照料 一 切 。 我 们 的 程序 
可 在 需要 的 时 候 创建 一 个 对 象 。 而 且 更 进一步 地 ， 根 本 不 必 担 心 那个 对 
象 的 传输 机 制 的 细节 : 只 需 简 单 地 传递 句柄 即 可 。 有 些 时 候 ， 这 种 简化 
非常 有 价值 ， 但 另 一 些 时 候 却 显得 有 些 多 余 。 


可 从 两 个 方面 认识 这 一 机 制 的 缺点 : 


(1) 肯定 要 为 额外 的 内 存 管理 付出 效率 上 的 损失 《尽管 损失 不 大 ) ， 而 
且 对 于 运行 所 需 的 时 间 ， 总 是 存在 一 丝 不 确定 的 因素 《因为 在 内 存 不 够 
时 ， 芯 圾 收集 需 可 能 会 被 强制 采取 行动 ) 。 对 大 多 数 应 用 来 说 ， 优 点 显 
得 比 缺 点 重要 ， 而 且 部 分 对 时 间 要 求 非常 苛刻 的 段落 可 以 用 native 方 法 
写成 (参见 附录 A) 。 


(2) ”别名 处 理 : 有 时 会 不 愤 获 得 指 癌 同 一 个 对 象 的 两 个 句柄 。 只 有 在 这 
两 个 句柄 都 假定 指 同一 个 “明确 ”的 对 象 时 ， 才 有 可 能 产生 问题 。 对 这 个 
问题 ， 必 须 加 以 足够 的 重视 。 而 且 应 该 尽 可 能 地 “克隆 ”一 个 对 象 ， 以 防 
止 另 一 个 句柄 被 不 希望 的 改动 影响 。 除 此 以 外 ， 可 考虑 创建 “不 可 变 ” 对 
象 ， 使 它 的 操作 能 返回 同 种 类 型 或 不 同 种 类 型 的 一 个 新 对 象 ， 从 而 提高 
程序 的 执行 效率 。 但 干 万 不 要 改变 原始 对 象 ， 使 对 那个 对 象 别名 的 其 他 
任何 方面 都 感觉 不 出 变化 。 


有 些 人 认为 Java 的 克隆 是 一 个 笨拙 的 家 伙 ， 所 以 他 们 实现 了 自己 的 元 隆 
方案 (注释 @) ， 永 远 杜 绝 调用 Object.clone() 方 法 ， 从 而 消除 了 实现 
Cloneable 和 捕获 CloneNotSupportException 违 例 的 需要 。 这 一 做 法 是 合理 
的 ， 而 且 由 于 done() 在 Java 标 准 库 中 很 少 得 以 支持 ， 所 以 这 显然 也 是 一 
种 “安全 ”的 方法 。 只 要 不 调用 Object.clone()， 就 不 必 实 现 Cloneable 或 者 
捕获 违例 ， 所 以 那 看 起 来 也 是 能 够 接受 的 。 





















































©: Doug Lea 特 别 重 视 这 个 问题 ， 并 把 这 个 方法 推荐 给 了 我 ， 他 说 只 需 
为 每 个 类 都 创建 一 个 名 为 duplicateO 的 函数 即 可 。 


Java 中 一 个 有 趣 的 关键 字 是 byvalue 〈 按 值 ) ， 它 属于 那些 “保留 但 未 实 
现 ” 的 关键 字 之 一 。 在 理解 了 别名 和 克隆 问题 以 后 ， 大 家 可 以 想象 
byvalue 最 终 有 一 天 会 在 Java 中 用 于 实现 一 种 自动 化 的 本 地 副本 。 这 样 做 
Se 复杂 的 克隆 问题 ， 并 使 这 种 情况 下 的 编写 的 代码 变 得 更 加 
简单 和 健壮 。 





12.6 练习 


(1) 创建 一 个 myString 类 ， 在 其 中 包含 了 一 个 String 对 象 ， 以 便 用 在 构建 
器 中 用 构建 器 的 自 变量 对 其 进行 初始 化 。 添 加 一 个 toString0) 方 法 以 及 一 
个 concatenate() 方 法 ， 令 其 将 一 个 String 对 象 妃 加 到 我 们 的 内 部 字 串 。 在 
myString 中 实现 clone()。 创 建 两 个 static 方 法 ， 每 个 都 取得 一 个 myString x 
句柄 作为 自己 的 自 变 量 ， 并 调用 x.concatenate("test]。 但 在 第 二 个 方法 
中 ， 请 首先 调用 clone0)。 测 试 这 两 个 方法 ， 观 察 它 们 不 同 的 结 


(2) ”创建 一 个 名 为 Battery (电池) 的 类 ， 在 其 中 包含 一 个 int， 用 它 表示 
电池 的 编号 (采用 独一无二 的 标识 符 的 形式 ) 。 接 下 来 ， 创 建 一 个 名 为 
Toy 的 类 ， 其 中 包含 了 一 个 Battery 数 组 以 及 一 个 toString， 用 于 打印 出 所 
有 电池 。 为 Toy 写 一 个 cdlone() 方 法 ， 令 其 自动 关闭 所 有 Battery 对 象 。 元 

隆 Toy 并 打印 出 结果 ， 完 成 对 它 的 测试 。 


(3) 修改 CheckCloneable.java， 使 所 有 clone() 方 法 都 能 捕获 
CloneNotSupportException 违 例 ， 而 不 是 把 它 直接 传递 给 调用 者 。 

(4) 修改 Compete.java， 为 Thing2 和 Thing4 类 添加 更 多 的 成 员 对 象 ， 看 看 
自己 是 个 能 判断 计时 随 复杂 性 变化 的 规律 一 一 是 一 种 简单 的 线性 关系 ， 
还 是 看 起 来 更 加 复杂 。 


(5) 从 Snake.java 开 始 ， 创 建 Snake 的 一 个 深层 复制 版 本 。 








HSE 创建 窗口 和 程序 片 


在 Java 1.0 中 ， 图 形 用 户 接口 《GUI) 库 最 初 的 设计 目标 是 让 程序 员 构 建 
一 个 通用 的 GUI， 使 其 在 所 有 平台 上 都 能 正常 显示 。 


但 遗憾 的 是 ， 这 个 目标 并 未 达到 。 事 实 上，Java 1.0 版 的 “抽象 Windows 
工具 包 ”(AWT) 产生 的 是 在 各 系统 看 来 都 同样 欠 佳 的 图 形 用 户 接口 。 
除 此 之 外 ， 它 还 限制 我 们 只 能 使 用 四 种 字体 ， 并 且 不 能 访问 操作 系统 中 
现 有 的 高 级 GUI 元 素 。 同 时 ，Jave1.0 版 的 AWT 编 程 模 型 也 不 是 面向 对 象 
的 ， 极 不 成 熟 。 这 类 情况 在 Javal.1 版 的 AWT 事 件 模型 中 得 到 了 很 好 的 
改进 ， 例 如 :更 加 清晰 、 面 向 对 象 的 编程 、 遵 循 Java Beans 的 范例 ， 以 
及 一 个 可 轻松 创建 可 视 编 程 环境 的 编程 组 件 模 型 。Javal.2 为 老 的 Java 
1.0 AWT 添 加 了 Java 基 础 类 (‘AWT)， 这 是 一 个 被 称 为 “Swing” 的 GUI 的 
一 部 分 。 丰 富 的 、 易 于 使 用 和 理解 的 Java Beans 能 经 过 拖 放 操作 〈( 像 手 
工 编程 一 样 的 好 ) ， 创 建 出 能 使 程序 员 满 意 的 GUI。 软 件 业 的 “3 次 修订 
版 ?规则 看 来 对 于 程序 设计 语言 也 是 成 立 的 《一 个 产品 除非 经 过 第 3 次 修 
订 ， 否 则 不 会 尽 如 人 意 ) 。 


Java 的 主要 设计 目的 之 一 是 建立 程序 片 ， 也 就 是 建立 运行 在 WEB 浏览 
器 上 的 小 应 用 程序 。 由 于 它们 必须 是 安全 的 ， 所 以 程序 片 在 运行 时 必须 
加 以 限制 。 无 论 怎样 ， 它 们 都 是 文 持 客户 器 编程 的 强 有 力 的 工具 ， 一 个 
重要 的 应 用 便 是 在 Web 上 。 


在 一 个 程序 片 中 编程 会 受到 很 多 的 限制 ， 我 们 一 般 说 它 “ 在 沙 箱 内 ”， 这 
是 由 于 Java 运 行 时 一 直 会 有 某 个 东西 即 Java 运 行 期 安全 系统 TE 
监视 着 我 们 。Jave ”1.1 为 程序 片 提供 了 数字 签名 ， 所 以 可 选 出 能 信赖 的 
程序 片 去 访问 主机 。 不 过 ， 我 们 也 能 跳出 沙 箱 的 限制 写 出 可 靠 的 程序 。 
在 这 种 情况 下 ， 我 们 可 访问 操作 系统 中 的 其 他 功能 。 在 这 本 书 中 我 们 自 
始 至 终 编 写 的 都 是 可 靠 的 程序 ， 但 它们 成 为 了 没有 图 形 组 件 的 控制 台 程 
序 。AWT 也 能 用 来 为 可 靠 的 程序 建立 GUI 接口 。 


在 这 一 章 中 我 们 将 先 学 习 使 用 老 的 AWT 工 具 ， 我 们 会 与 许多 文 持 和 使 
用 AWT 的 代码 程序 样本 相遇 。 尽 管 这 有 一 些 困 难 ， 但 却 是 必须 的 ， 
为 我 们 必须 用 老 的 AWT 来 维护 和 阅读 传统 的 Java 人 代码。 有 时 甚 全 需要 我 
们 编写 AWT 代 码 去 支持 不 能 从 Javal.0 升 级 的 环境 。 在 本 章 第 二 部 分 ， 
我 们 将 学 习 Java 1.1 版 中 新 的 AWT 结 构 并 会 看 到 它 的 事件 模型 是 如 此 的 





























优秀 (如 果 能 掌握 的 话 ， 那 么 在 编制 新 的 程序 时 就 可 使 用 这 最 新 的 工 
具 。 最 后 ， 我 们 将 学 习 新 的 能 像 类 库 一 样 加 入 到 Java 1.1 版 中 的 
JFC/Swing 组 件 ， 这 意味 着 不 需要 升级 到 Java 1.2 便 能 使 用 这 一 类 库 。 


大 多 数 的 例 程 都 将 展示 程序 片 的 建立 ， 这 并 不 仅仅 是 因为 这 非常 的 容 

易 ， 更 因为 这 是 AWT 的 主要 作用 。 另 外 ， 当 用 AWT 创 建 一 个 可 靠 的 程 
序 时 ， 我 们 将 看 到 处 理 程 序 的 不 同 之 处 ， 以 及 怎样 创建 能 在 命令 行 和 浏 
览 髓 中 运行 的 程序 。 


请 注意 的 是 这 不 是 为 了 摘 述 类 的 所 有 程序 的 综合 解释 。 这 一 章 将 带领 我 
们 从 摘要 开始 。 当 我 们 查找 更 复杂 的 内 容 时 ， 请 确定 我 们 的 信息 浏览 器 
通过 查找 类 和 方法 来 解决 编程 中 的 问题 (如 果 我 们 正在 使 用 一 个 开发 环 
境 ， 信 息 浏览 器 也 许 是 内 建 的 ， 如 果 我 们 使 用 的 是 SUN 公 司 的 JDK 则 这 
时 我 们 要 使 用 WEB 浏 览 器 并 在 Java 根 目录 下 面 开 始 ) 。 附 录 F 列 出 了 用 
于 深入 学 习 库 知识 的 其 他 一 些 参 考 资料 。 

















13.1 为 何 要 用 AWT? 


对 于 本 章 要 学 习 的 “老式 "AWT， 它 最 严重 的 缺点 就 是 它 无 论 在 面 辣 对 象 
设计 方面 ， 还 是 在 GUI 开 发 包 设 计 方 面 ， 都 有 不 尽 如 人 意 的 表现 。 它 使 
我 们 回 到 了 程序 设计 的 黑暗 年 代 〈 换 成 其 他 话 就 是 “拙劣 的 “可 和 
的 “恶劣 的 等 等 ) 。 必 须 为 执行 每 一 个 事件 编写 代码 ， 包 括 在 其 他 
环境 中 利用 “资源 ” 即 可 轻松 完成 的 一 些 任务 。 


许多 象 这 样 的 问题 在 Java 1.1 里 都 得 到 了 绥 解 或 排除 ， 因 为 : 


(1)Java 1.1 的 新 型 AWT 是 一 个 更 好 的 编程 模型 ， 并 同 更 好 的 库 设计 迈 出 
了 可 喜 的 一 步 。 而 Java Beans 则 是 那个 库 的 框架 。 


(2)“GUI 构 建 占 *”( 可 视 编程 环境 ) 将 适用 于 所 有 开发 系统 。 在 我 们 用 图 

形 化 工具 将 组 件 置 入 窗 体 的 时 候 ，Java Beans 和 新 的 AWT 使 GUI 构 建 器 

Bn le EN eae eee eon 
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既然 如 此 ， 为 什么 还 要 学 习 使 用 老 的 AWT 呢 ?原因 很 简单 ， 因 为 它 的 
存在 是 个 事实 。 就 目前 来 说 ， 这 个 事实 对 我 们 来 说 显得 有 些 不 利 ， 它 涉 
及 到 面向 对 象 库 设 计 的 一 个 宗旨: 一 旦 我 们 在 库 中 公布 一 个 组 件 ， 就 再 
不 能 去 掉 它 。 如 去 掉 它 ， 束 会 损害 别人 已 存在 的 代码 。 另 外 ， 当 我 们 学 
习 Java 和 所 有 使 用 老 AWT 的 程序 时 ， 会 发 现 有 许多 原来 的 代码 使 用 的 都 
是 老式 AWT。 


AWT 必 须 能 与 固有 操作 系统 的 GUI 组 件 打 交通 ， 这 意味 着 它 需 要 执行 一 
个 程序 片 不 可 能 做 到 的 任务 。 一 个 不 被 信任 的 程序 片 在 操作 系统 中 不 能 
作出 任何 直接 调用 ， 人 否则 和 它 会 对 用 户 的 机 器 做 出 不 恰当 的 事情 。 一 个 不 
被 信任 的 程序 片 不 能 访问 重要 的 功能 。 例 如 , “在 屏幕 上 男 一 个 窗口 ”的 
唯一 方法 是 通过 调用 拥有 特殊 接口 和 安全 检查 的 标准 Java 库 。Sun 公 司 
的 原始 模型 创建 的 信任 库 将 仅仅 供给 web 浏览 右 中 的 Java 系 统 信任 关系 
目 动 授权 右 使 用 ， 自 动 授 权 右 将 控制 怎样 进入 到 库 中 去 。 


但 当 我 们 想 增 加 操作 系统 中 访问 新 组 件 的 功能 时 该 怎么 办 ? 等待 Sun 来 
决定 我 们 的 扩展 被 合并 到 标准 的 Java 库 中 ， 但 这 不 一 定 会 解决 我 们 的 问 
题 。Java 1.1 版 中 的 新 模型 是 “信任 代码 ”或 “签名 代码 ”因此 一 个 特殊 服 
































务 圳 将 校 验 我 们 下 载 的 、 由 规定 的 开发 者 使 用 的 公共 密 钥 加 密 系统 的 代 
码 。 这 样 我 们 束 可 知道 代码 从 何 而 来 ， 那 真 的 是 Bob 的 代码 ， 还 是 由 系 
人 伪装 成 Bob 的 代码 。 这 并 不 能 阻止 Bob 犯 错误 或 作 某 些 恶 意 的 事 ， 但 
能 防止 Bob 逃 避 匿 名 制造 计算 机 病毒 的 责任。 一 个 数字 签名 的 程序 片 
“被 信任 的 程序 片 盖 一 在 Java 1.1 版 能 进入 我 们 的 机 器 并 直接 控制 
它 ， 正 像 一 些 其 它 的 应 用 程序 从 信任 关系 自动 授权 机 中 得 到 “信任 ”并 安 
装 在 我 们 的 机 器 上 。 


这 是 老 AWT 的 所 有 特点 。 老 的 AWT 代 码 将 一 直 存 在 ， 新 的 Java 编 程 者 
在 从 旧 的 书本 中 学 习 时 将 会 遇 到 老 的 AWT 代 码 。 同 样 ， 老 的 AWT 也 是 
值得 去 学 习 的 ， 例 如 在 一 个 只 有 少量 库 的 例 程 设计 中 。 老 的 AWT 所 包 
括 的 范围 在 不 考虑 深度 和 枚 举 每 一 个 程序 和 类 ， 取 而 代 之 的 是 给 了 我 们 
一 个 老 AWT 设 计 的 概貌 。 














13.2 基本 程序 所 


库 通 常 按 照 它 们 的 功能 来 进行 组 合 。 一 些 库 ， 例 如 使 用 过 的 ， 便 中 断 搁 
置 起 来 。 标 准 的 Java 库 字符 串 和 矢量 类 就 是 这 样 的 一 个 例子 。 其 他 的 库 
被 特殊 地 设计 ， 例 如 构建 块 去 建立 其 它 的 库 。 库 中 的 某 些 类 是 应 用 程序 
的 框架 ， 其 目的 是 协助 我 们 构建 应 用 程序 ， 在 提供 类 或 类 集 的 情况 下 产 
生 每 个 特定 应 用 程序 的 基本 活动 状况 。 然 后 ， 为 我 们 定制 活动 状况 ， 必 
须 继承 应 用 程序 类 并 且 上 废弃 程序 的 权益 。 应 用 程序 框架 的 默认 控制 结构 
将 在 特定 的 时 间 调 用 我 们 废弃 的 程序 。 应 用 程序 的 框架 是 “分 离 、 改 变 
a ee E ete 
殊 程 序 段 。 


























程序 片 利用 应 用 程序 框架 来 建立 。 我 们 从 类 中 继承 程序 片 ， 并 且 废 弃 特 
定 的 程序 。 大 多 数 时 间 我 们 必须 考虑 一 些 不 得 不 运行 的 使 程序 片 在 
WEB 页 面 上 建立 和 使 用 的 重要 方法 。 这 些 方法 是 : 





init( ) Called when the applet is first created to perform first-time 
initialization of the applet 


start() Called every time the applet moves into sight on the Web 
browser to allow the applet to start up its normal operations 
(especially those that are shut off by stop( )). Also called after 


init( ). 


paint() liPart of the base class Component (three levels of inheritance 
up). Called as part of an update( ) to perform special painting on 


the canvas of an applet. 


stop() {Called every time the applet moves out of sight on the Web 


| | 


browser to allow the applet to shut off expensive operations. Also 
called right before destroy( ). 





destroy( )|Called when the applet is being unloaded from the page to 
perform final release of resources when the applet is no longer 
used 


方法 作用 
init) 程序 片 第 一 次 被 创建 ， 初 次 运行 初始 化 程序 片 时 调用 


start) 每 当 程 序 片 进入 Web 浏 览 嚣 中， 并 且 人 允许 程序 片 启动 它 的 常规 操 
作 时 调用 特殊 的 程序 片 被 stop0 关 闭 〉; 同样 在 init0 后 调用 


paint() 基础 类 Component 的 一 部 分 (继承 结构 中 上 济 三 级 ) 。 作 为 
update0 的 一 部 分 调用 ， 以 便 对 程序 片 的 画布 进行 特殊 的 描绘 


stop) ”每 次 程序 片 从 Web 浏 览 器 的 视线 中 离开 时 调用 ， 使 程序 片 能 关闭 
代价 高 昂 的 操作 ;同样 在 调用 destroy0O 前 调用 


destroy) 程序 片 不 再 需要 ， 将 它 从 页 面 中 秋 载 时 调用 ， 以 执行 资源 的 最 
后 清除 工作 


现在 来 看 一 看 paint0) 方 法 。 一 旦 Component (目前 是 程序 片 ) 决定 自己 
需要 更 新 ， 就 会 调用 这 个 方法 可 能 是 由 于 它 再 次 回转 屏幕 ， 首 次 在 
屏幕 上 显示 ， 或 者 是 由 于 其 他 窗口 临时 履 盖 了 你 的 web 浏览 器 。 此 时 程 
序 片 会 调用 它 的 update(0) 方 法 〈 在 基础 类 Component 中 定义 ) ， 访 方法 会 
恢复 一 切 该 恢复 的 东西 ， 而 调用 paintO 正 是 这 个 过 程 的 一 部 分 。 没 必要 
对 paintO 进 行 过 载 处 理 ， 但 构建 一 个 简单 的 程序 片 无 疑 是 方便 的 方法 ， 
所 以 我 们 首先 从 paint() 方 法 开始 。 


updateO 调 用 paintO0 时 ， 会 回 其 传递 指 癌 Graphics 对 象 的 一 个 句柄 ， 那 个 
WARNES TELM ERD 的 表面 。 这 是 非常 重要 的 ， 因 为 我 们 
受到 项 目 组 件 的 外 观 的 限制 ， 因 此 不 能 画 到 区 域外 ， 这 可 是 一 件 好 事 ， 
| 在 程序 片 的 例子 中 ， 程 序 片 的 外 观 就 是 这 界 
cE HJ LX EX o 


图 形 对 象 同样 有 一 系列 我 们 可 对 其 进行 的 操作 。 这 些 操作 都 与 在 画布 上 
作 图 有 关 。 上 所 以 其 中 的 大 部 分 都 要 涉及 图 像 、 几 何 沫 状 、 圆 踊 等 等 的 描 
绘 〈 注 意 如 果 有 兴趣 ， 可 在 Java 文 档 中 找到 更 详细 的 说 明 ) 。 有 些 方法 


























允许 我 们 男 出 字符 ， 而 其 中 最 常用 的 就 是 drawString0。 对 于 它 ， 需 指出 
目 己 想 描绘 的 String《〈 字 串 ) ， 并 指定 它 在 程序 片 作 图 区 域 的 起 点 。 这 
个 位 置 用 像素 表示 ， 所 以 它 在 不 同 的 机 器 上 看 起 来 是 不 同 的 ， 但 至 少 是 
可 以 移植 的 。 


根据 这 些 信息 即 可 创建 一 个 简单 的 程序 片 : 


//: Applet1.java 


// Very simple applet 

package c13; 

import java.awt.*; 

import java.applet.*; 

public class Applet1 extends Applet { 
public void paint(Graphics g) { 

g.drawString("First applet", 10, 10); 

} 

} ///:~ 





注意 这 个 程序 片 不 需要 有 一 个 main()。 所 有 内 容 都 封装 到 应 用 程序 框架 
中 ; 我 们 将 所 有 启动 代码 都 放 在 init() 里 。 

必须 将 这 个 程序 放 到 一 个 web 页 中 才能 运行 ， 而 只 能 在 文 持 Java 的 Web 
浏览 器 中 才能 看 到 此 页 。 为 了 将 一 个 程序 片 置 入 Web 页 ， 需 要 在 那个 
Web 页 的 代码 中 设置 一 个 特殊 的 标记 【注释 四 ) ， 以 指示 网 页 装载 和 运 
行程 序 片 。 这 就 是 applet 标 记 ， 它 在 Applet1 中 的 样子 如 下 : 


<applet 


code=Applet1i 


width=200 
height=200> 


</applet> 


QO: 本 书 假定 读者 已 掌握 了 HTML 的 基本 知识 。 这 些 知识 不 难 学 习 ， 有 
许多 书籍 和 网 上 资源 都 可 以 提供 帮助 。 


其 中 ，code 值 指定 了 .class 文 件 的 名 字 ， 程 序 片 就 驻 留 在 那个 文件 中 。 
width 和 height 指 定 这 个 程序 请 的 初始 尺寸 如 前 所 述 ， 以 像素 为 单 

KL) 。 还 可 将 另 一 些 东西 放 入 applet 标 记 : 用 于 在 因特网 上 寻找 其 

他 .class 文 件 的 位 置 (codebase) 、 对 齐 和 排列 信息 Calign) 、 使 程序 片 
相互 间 能 够 通信 的 一 个 特殊 标识 符 (name) 以 及 用 于 提供 程序 片 能 接收 
的 信息 的 参数 。 参 数 采 取 下 述 形式 : 








<Paramname= 标 识 符 value =" 信 息 "> 
可 根据 需要 设置 任意 多 个 这 样 的 参数 。 


在 简单 的 程序 片 中 ， 我 们 要 做 的 唯一 事情 是 按 上 述 形 式 在 Web 页 中 设置 
一 个 程序 片 标记 Capplet) ， 令 其 装载 和 运行 程序 片 。 


13.2.1 程序 片 的 测试 


我 们 可 在 不 必 建 立 网 络 连接 的 前 提 下 进行 一 次 简单 的 测试 ， 方 法 是 启动 
我 们 的 Web 浏 览 器 ， 然 后 打开 包含 了 程序 片 标签 的 HTML 文 件 (Sun 公 
司 的 JDK 同 样 包括 一 个 称 为 “程序 片 观察 器 ”的 工具 ， 它 能 挑 出 html 文 件 
的 <applet> 标 记 ， 并 运行 这 个 程序 片 ， 不 必 显 示 周 围 的 HTML 文 本 一 一 
IFO) 。html 文 件 载 入 后 ， 浏 览 器 会 发 现 程 序 片 的 标签 ， 并 查找 由 
code 值 指定 的 .class 文 件 。 当 然 ， 它 会 先 在 CLASSPATH (类 路 径 ) 中 寻 
找 ， 如 果 在 CLASSPATH 下 找 不 到 类 文件 ， 就 在 WEB 浏 览 器 状态 栏 给 出 
一 个 错误 信息 ， 告 知 不 能 找到 .class 文 件 。 


D; 由 于 程序 片 观察 器 会 忽略 除 APPLET 标 记 之 外 的 任何 东西 ， 所 以 可 
将 那些 标记 作为 注释 置 入 Java 源 人 码 : 

















// <applet code=MyApplet.class width=200 height=100></applet> 





这 样 就 可 直接 执行 “appletviewer MyApplet.java”， 不 必 再 创建 小 的 HTML 
文件 来 完成 测试 。 


若 想 在 Web 站 点 上 试验 ， 还 会 磁 到 另 一 些 麻 烦 。 首 先 ， 我 们 必须 有 一 个 
Web 站 点 ， 这 对 大 多 数 人 来 说 都 意味 着 位 于 远程 地 点 的 一 家 服务 提供 商 
CISP) 。 然 后 必须 通过 某 种 途径 将 HTML 文件 和 .class 文 件 从 自己 的 站 
点 移 至 ISP 机 器 上 正确 的 目录 “(WWW 目录 ) 。 这 一 般 是 通过 采用 “文件 
传输 协议 ”(ETP) 的 程序 来 做 成 的 ， 网 上 可 找到 许多 这 样 的 免费 程序 。 
所 以 我 们 要 做 的 全 部 事情 似乎 就 是 用 FTP 协 议 将 文件 移 至 ISP 的 机 器 ， 然 
后 用 自己 的 浏览 器 连接 网 站 和 HTML 文 件 ， 假如 程序 片 正 确 装 载 和 执 

行 ， 就 表明 大 功 告 成 。 但 真是 这 样 吗 ? 


但 这 儿 我 们 可 能 会 受到 愚弄 。 假 如 Web 浏 览 器 在 服务 器 上 找 不 到 .class 文 
件 ， 就 会 在 你 的 本 地 机 器 上 搜寻 CLASSPATH。 所 以 程序 片 或 许 根本 不 
能 从 服务 器 上 正确 地 装载 ， 但 在 你 看 来 却 是 一 切 正常 的 ， 因 为 浏览 右 在 
你 的 机 器 上 找到 了 它 需 要 的 东西 。 但 在 其 他 人 访问 时 ， 他 们 的 浏览 器 就 
无 法 找到 那些 类 文件 。 所 以 在 测试 时 ， 必 须 确 定 已 从 自己 的 机 器 删除 了 
相关 的 .class 文 件 ， 以 确保 测试 结果 的 真实 。 

我 自己 就 遇 到 过 这 样 的 一 个 问题 。 当 时 是 将 程序 片 置 入 一 个 
package〈 包 ) 中 。 上 载 了 HTML 文件 和 程序 片 后 ， 由 于 包 名 的 问题 ， 程 
序 片 的 服务 器 路 径 似乎 陷入 了 混乱 。 但 是 ， 我 的 浏览 器 在 本 地 类 路 径 
(CLASSPATH) 中 找到 了 它 。 这 样 一 来 ， 我 就 成 了 能 够 成 功 装 载 程序 
片 的 唯一 一 个 人 。 后 来 我 花 了 一 些 时 间 才 发 现 原来 是 package 语 句 有 
误 。 一 般 地 ， 应 该 将 package 语 句 置 于 程序 片 的 外 部 。 

13.2.2 一 个 更 图 形 化 的 例子 

这 个 程序 不 会 太 令 人 紧张 ， 所 以 让 我 们 试 着 增加 一 些 有 趣 的 图 形 组 件 。 


//: Applet2.java 























// Easy graphics 
import java.awt.*; 


import java.applet.*; 


public class Applet2 extends Applet { 
public void paint(Graphics g) { 
g.drawString("Second applet", 10, 15); 
g.draw3DRect(0, 0, 100, 20, true); 
} 
} ///:~ 








这 个 程序 用 一 个 方 框 将 字符 串 包 围 起 来 。 当 然 ， 所 有 数字 部 是 “ 便 编 
码 ” 的 《指数 字 固 定 于 程序 内 部 ) ， 并 以 像素 为 基础 。 所 以 在 一 些 机 器 
上 ， 框 会 正好 将 字 串 围 住 ， 而 在 另 一 些 机 器 上 ， 也 许 根本 看 不 见 这 个 
框 ， 因 为 不 同 机 器 安装 的 字体 也 会 有 所 区 别 。 


对 Graphic 类 而 言 ， 可 在 帮助 文档 中 找到 男 一 些 有 趣 的 内 容 。 大 多 数 涉 
及 图 形 的 活动 都 是 很 有 趣 的 ， 所 有 我 将 更 多 的 试验 留 给 读者 自己 去 进 
行 。 

13.2.3 框架 方法 的 演示 


观看 框架 方法 的 实际 运作 是 相当 有 趣 的 (这 个 例子 只 使 用 init()，start() 
和 和 stop()， 因 为 paint() 和 destroy() 非 常 简 单 ， 很 容易 就 能 掌握 。 下 面 的 
程序 片 将 跟踪 这 些 方 法 调用 的 次 数 ， 并 用 paint0) 将 其 显示 出 来 : 


//: Applet3.java 

















// Shows init(), start() and stop() activities 
import java.awt.*; 
import java.applet.*; 
public class Applet3 extends Applet { 
String s; 


int inits = 0; 


int starts = 0; 

int stops = 0; 

public void init() { inits++; } 
public void start() { startst++; } 
public void stop() { stops++; } 


public void paint(Graphics g) { 


s = "inits: " + inits + 
", starts: " + starts + 
", stops: " + stops; 
g.drawString(s, 10, 10); 
} 
} ///:~ 











正常 情况 下 ， 妆 我 们 过 载 一 个 方法 时 ， 需 检查 目 己 是 否 需 要 调用 方法 的 
基础 类 版 本 ， PRE Sed 例如 ， 使 用 init0 时 可 能 需要 调用 
super.init0。 然 而 ，Applet 文 档 特 别 指出 initO0、start0 和 stopO 在 Applet 中 
没有 用 处 ， 所 以 这 里 不 需要 调用 它们 。 


试验 这 个 程序 片 时 ， 会 发 现 假如 最 小 化 WEB 浏 览 器 ， 或 者 用 另 一 个 窗 
口 将 其 覆盖 ， 那 么 就 不 能 再 调用 stop0 和 start0) 〈 这 一 行为 会 随 着 不 同 的 
实现 方案 变化 ; 可 考 谍 将 Web 浏 览 器 的 行为 同 程序 三 观察 囊 的 行为 对 申 
一 下 ) 。 调 用 唯一 发 生 的 场合 是 在 我 们 转移 到 一 个 不 同 的 Web 页 ， 然后 
返回 包含 了 程序 片 的 那个 页 时 。 





13.3 制作 按钮 


制作 一 个 按钮 非常 简单 ， 只 需要 调用 Button 构 建 桌 ， 并 指定 想 在 按钮 上 
出 现 的 标签 束 行 了 如 果 不 想 要 标签 ， 亦 可 使 用 默认 构建 种 ， 但 那 种 情 
况 极 少 出 现 ) 。 可 参照 后 面 的 程序 为 按钮 创建 一 个 句柄 ， 以 便 以 后 能 够 
引 0 


Button 是 一 个 组 件 ， 象 它 目 己 的 小 窗口 一 样 ， 会 在 更 新 时 得 以 重 绘 。 这 
意味 着 我 们 不 必 明 确 描绘 一 个 按钮 或 者 其 他 任意 种 类 的 控件 ;只 需 将 它 
们 纳入 窗 体 ， 以 后 的 描绘 工作 会 由 它们 目 行 负 责 。 所 以 为 了 将 一 个 按钮 
置 入 窗 体 ， 需 要 过 载 init( 方 法 ， 而 不 是 过 载 paint0: 


//: Button1.java 


// Putting buttons on an applet 
import java.awt.*; 
import java.applet.*; 
public class Button1 extends Applet { 
Button 
b1 = new Button("Button 1"), 
b2 = new Button("Button 2"); 
public void init() { 
add(b1); 


add(b2); 


} ///:~ 


但 这 还 不 足以 创建 Button (或 其 他 任何 控件 ) 。 必 须 同时 调用 Applet 
add0 方 法 ， 令 按钮 放置 在 程序 片 的 窗 体 中 。 这 看 起 来 似乎 比 实际 简单 得 
多 ， 因 为 对 add0 的 调用 实际 会 〈 间 接地 ) 决定 将 控件 放 在 窗 体 的 什么 地 
方 。 对 窗 体 布局 的 控件 号 上 就 要 讲 到 。 





13.4 捕获 事件 


大 家 可 注意 到 假如 编译 和 运行 上 面 的 程序 片 ， 按 下 按钮 后 不 会 发 生 任 何 
事情 。 必 须 进 入 程序 片 内 部 ， 编 写 用 于 决定 要 发 生 什么 事情 的 代码 。 对 
于 由 事件 驱动 的 程序 设计 ， 它 的 基本 目标 束 是 用 代码 捕获 发 生 的 事件 ， 
并 由 代码 对 那些 事件 作出 啊 应 。 事 实 上 ，GUI 的 大 部 分 内 容 部 是 围绕 这 
种 事件 驱动 的 程序 设计 展开 的 。 


经 过 本 书 前 面 的 学 习 ， 大 家 应 该 有 了 面 癌 对 象 程序 设计 的 一 些 基础 ， 此 
时 可 能 会 想到 应 当 有 一 些 面 向 对 象 的 方法 来 专门 控制 事件 。 例 如 ， 也 许 
不 得 不 继承 每 个 按钮 ， 并 过 载 一 些 “ 按 钮 按 下 "方法 〈 尽 管 这 显得 非常 麻 
MAAR) 。 大 家 也 可 能 认为 存在 一 些 主 控 “ 事 件 ? 类 ， 其 中 为 希望 啊 应 
的 每 个 事件 都 包含 了 一 个 方法 。 


在 对 象 以 前 ， 事 件 控制 的 典型 方式 是 switch 语 句 。 每 个 事件 都 对 应 一 个 
独一无二 的 整数 编写 ;而且 在 主事 件 控制 方法 中 ， 需 要 专门 为 那个 值 写 


一 个 Switch 。 


Java 1.0 的 AWT 没 有 采用 任何 面 癌 对 象 的 手段 。 此 外 ， 它 也 没有 使 用 
switch 语 句 ， 没 有 打算 依靠 那些 分 配给 事件 的 数字 。 相 反 ， 我 们 必须 创 
建 f 语 句 的 一 个 般 套 系列 。 通 过 if 语 句 ， 我 们 需要 尝试 做 的 事情 是 侦 测 

到 作为 事件 “目标 ”的 对 象 。 换 言 之 ， 那 是 我 们 关心 的 全 部 内 容 一 一 假如 
某 个 按钮 是 一 个 事件 的 目标 ， 那 么 它 肯 定 是 一 次 鼠标 点 击 ， 并 要 基于 那 
个 假设 继续 下 去 。 但 是 ， 事 件 里 也 可 能 包含 了 其 他 信息 。 例 如 ， 假 如 想 
调查 一 次 鼠标 点 击 的 像素 位 置 ， 以 便 画 一 条 引 癌 那个 位 置 的 线 ， 那 么 

Event 对 象 里 就 会 包含 那个 位 置 的 信息 〈 也 要 注意 Java 1.0 的 组 件 只 能 产 
而 Java 1.1 和 Swing/JFC 组 件 则 可 产生 完整 的 一 系列 

Jos 


Java 1.0 版 的 AWT 方 法 串联 的 条 件 语句 中 存在 action0) 方 法 的 调用 。 虽 然 
整个 Java 1.0 版 的 事件 模型 不 兼容 Java 1.1 版 ， 但 它 在 还 不 文 持 Javal.1 版 
的 机 器 和 运行 简单 的 程序 片 的 系统 中 更 广泛 地 使 用 ， 上 忠告 您 使 用 它 会 变 
得 非常 的 舒适 ， 包 括 对 下 面 使 用 的 action0) 程 序 方法 而 言 。 


acion) HA ATARE: 第 一 个 是 事件 的 类 型 ， 包 括 所 有 的 触发 调用 
action0 的 事件 的 有 关 信 息 。 例 如 鼠标 单 击 、 普 通 按键 按 下 或 释放 、 特 殊 















































按键 按 下 或 释放 、 鼠 标 移动 或 者 拖 动 、 事 件 组 件 得 到 或 丢失 焦点 ， 等 
等 。 第 二 个 目 变 量 通常 是 我 们 忽略 的 事件 目标 。 第 二 个 上 自 变 量 封装 在 事 
件 目标 中 ， 所 以 它 像 一 个 自 变 量 一 样 的 元 长 。 


需 调 用 action0 时 情况 非常 有 限 : 将 控件 置 入 窗 体 时 ， 一 些 类 型 的 控件 

GE BWE FAIRE, KA) 会 发 生 一 种 “标准 行动 "从 而 随 
相应 的 Event 对 象 发 起 对 action() 的 调用 。 比 如 对 按钮 来 说 ， 一 旦 按钮 被 
按 下 ， 而 且 没 有 再 多 按 一 次 ， 束 会 调用 它 的 action0 方 法 。 这 种 行为 通常 
正 是 我 们 所 希望 的 ， 因 为 这 正 是 我 们 对 一 个 按钮 正常 观感 。 但 正如 本 章 
yee eee 还 可 通过 handleEvent() 方 法 来 处 理 其 他 许多 类 型 的 


前 面 的 例 程 可 进行 一 些 扩展 ， 以 便 象 下 面 这 样 控制 按钮 的 点 击 : 


//: Button2.java 














// Capturing button presses 
import java.awt.*; 
import java.applet.*; 
public class Button2 extends Applet { 
Button 
b1 = new Button("Button 1"), 
b2 = new Button("Button 2"); 
public void init() { 
add(b1); 
add(b2); 
} 
public boolean action(Event evt, Object arg) { 


if(evt.target.equals(b1)) 


getAppletContext().showStatus("Button 1"); 
else if(evt.target.equals(b2) ) 
getAppletContext().showStatus("Button 2"); 
// Let the base class handle it: 
else 
return sSuper.action(evt, arg); 
return true; // We've handled it here 


} 
SS 


为 了 解 目标 是 什么 ， 需 要 同 Event 对 象 询问 它 的 target( 目 标 〉 成 员 是 什 
2 然后 用 equals0) 方 法 检查 它 是 侣 与 日 己 感 兴趣 的 目标 对 象 名 柄 相 

。 为 所 有 感 兴趣 的 对 象 写 好 句柄 后 ， 必 须 在 末尾 的 else 语 句 中 调用 
ww arg) 方 法 。 我 们 在 第 7 章 已 经 说 过 〈《 有 关 多 形 性 的 那 一 
eH) ， 此 时 调用 的 是 我 们 过 载 过 的 方法 ， 而 非 它 的 基础 类 版 本 。 然 而 ， 
基础 类 版 本 也 针对 我 们 不 感 兴趣 的 所 有 情况 提供 了 相应 的 控制 代码 。 除 
非 明确 进行 ， 人 否则 它们 是 不 会 得 到 调用 的 。 返 回 值 指出 我 们 是 售 己 经 处 
T es a 就 应 返回 true; 否则 就 返回 由 
基础 类 event(0) 返 回 的 东西 。 


ANATA EAR Be 最 简单 的 行动 就 是 打印 出 到 底 是 什么 按钮 被 按 下 。 

些 系统 允许 你 弹出 一 个 小 消息 窗口 ， 但 Java 程 序 片 却 防 本 时 窗口 的 弹出 。 
不 过 我 们 可 以 用 调用 Applet 方 法 的 getAppletContext() 来 访问 浏览 器 ， 然 
后 用 showStatus() 在 浏览 器 窗口 底部 的 状态 栏 上 显示 一 条 信息 (注释 
©) 。 还 可 用 同样 的 方法 打印 出 对 事件 的 一 段 完整 说 明文 字 ， 方法 是 调 
用 getAppletConext().showStatus(evt + "。 空 字 串 会 eae 
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®©: ShowStatus0) 也 属于 Applet 的 一 个 方法 ， 所 以 可 直接 调用 它 ， 不 必 调 
用 getAppletContext()。 




















尽管 看 起 来 似乎 很 奇怪 ， 但 我 们 确实 也 E 通 过 eventO 中 的 第 二 个 参数 将 
一 个 事件 与 按钮 上 的 文字 相配 。 采 用 这 种 方法 ， 上 面 的 例子 就 变 成 了 


//: Button3.java 


// Matching events on button text 
import java.awt.*; 

import java.applet.*; 

public class Button3 extends Applet { 


Button 


b1 new Button("Button 1"), 


b2 = new Button("Button 2"); 


public void init() { 
add(b1); 
add(b2); 
} 
public boolean action (Event evt, Object arg) { 
if(arg.equals("Button 1")) 
getAppletContext().showStatus("Button 1"); 
else if(arg.equals("Button 2")) 
getAppletContext().showStatus("Button 2"); 
// Let the base class handle it: 
else 


return Super.action(evt, arg); 


return true; // We've handled it here 
} 


} ///:~ 


很 难 确切 知道 equals() 方 法 在 这 儿 要 做 什么 。 这 种 方法 有 一 个 很 大 的 问 
题 ， 就 是 开始 使 用 这 个 新 技术 的 Java 程 序 员 至 少 需要 花费 一 个 受挫 折 的 
时 期 来 在 比较 按钮 上 的 文字 时 发 现 他 们 要 么 大 写 了 要 么 写 错 了 【我 束 有 
这 种 经 验 ) 。 同 样 ， 如 果 我 们 改变 了 按钮 上 的 文字 ， 程 序 代码 将 不 再 工 
作 “ 但 我 们 不 会 得 到 任何 编译 时 和 运行 时 的 信息 ) 。 所 以 如 果 可 能 ， 我 
们 就 得 避免 使 用 这 种 方法 。 








13.5 文本 字段 


“文本 字段 "是 允许 用 户 输 入 和 编辑 文字 的 一 种 线性 区 域 。 文 本 字段 从 文 
本 组 件 那 里 继承 了 让 我 们 选择 文字 、 让 我 们 像 得 到 字符 串 一 样 得 到 选择 
的 文字 ， 得 到 或 设置 文字 ， 设 置 文本 字段 是 否 可 编辑 以 及 连同 我 们 从 在 
线 参 考 书 中 找到 的 相关 方法 。 下 面 的 例子 将 证 明文 本 字段 的 其 它 功 能 ; 
我 们 能 注意 到 方法 名 是 显而易见 的 : 


//: TextField1.java 











// Using the text field control 
import java.awt.*; 
import java.applet.*; 
public class TextField1i extends Applet { 
Button 
b1 = new Button("Get Text"), 
b2 = new Button("Set Text"); 
TextField 
t = new TextField("Starting text", 30); 
String s = new String(); 
public void init() { 
add(b1); 
add(b2); 
add(t); 
} 


public boolean action (Event evt, Object arg) { 


if(evt.target.equals(b1)) { 
getAppletContext().showStatus(t.getText()); 
s = t.getSelectedText(); 
if(s.length() == 0) s = t.getText(); 
t.setEditable(true); 
} 
else if(evt.target.equals(b2)) { 
t.setText("Inserted by Button 2: " + s); 
t.setEditable(false); 
} 
// Let the base class handle it: 
else 
return super.action(evt, arg); 
return true; // We've handled it here 
} 
} ///:~ 





有 几 种 方法 均 可 构建 一 个 文本 字段 ， 其 中 之 一 是 提供 一 个 初始 字符 串 ， 
并 设置 字符 域 的 大 小 。 


按 下 按钮 1 是 得 到 我 们 用 鼠标 选择 的 文字 就 是 得 到 字段 内 所 有 的 文字 并 
转换 成 字符 串 S。 它 也 允许 字段 被 编辑 。 按 下 按钮 2 ” 放 一 条 信息 和 字符 
串 s 到 Text fields， 并 且 阻 止 字段 被 编辑 〈 尽 管 我 们 能 够 一 直选 择 文 
字 ) 。 文 字 的 可 编辑 性 是 通过 setEditable() 的 真 假 值 来 控制 的 。 





13.6 文本 区 域 


“文本 区 域 ” 很 像 文 字 字 段 ， 只 是 它 拥 有 更 多 的 行 以 及 一 些 引 人 注目 的 更 
多 的 功能 。 另 外 你 能 在 给 定位 置 对 一 个 文本 字段 退 加 、 插 入 或 者 修改 文 
字 。 这 看 起 来 对 文本 字段 有 用 的 功能 相当 不 错 ， 所 以 设法 发 现 它 设 计 的 
特性 会 产生 一 些 困 惑 。 我 们 可 以 认为 如 果 我 们 处 处 需要 “文本 区 域 ” 的 功 
能 ， 那 么 可 以 简单 地 使 用 一 个 线 型 文字 区 域 在 我 们 将 另外 使 用 文本 字段 
的 地 方 。 在 Java 1.0 版 中 ， 当 它们 不 是 固定 的 时 候 我 们 也 得 到 了 一 个 文 
本 区 域 的 垂直 和 水 平方 向 的 滚动 条 。 在 Java 1.1 版 中 ， 对 高 级 构建 器 的 
修改 允许 我 们 选择 哪个 深 动 条 是 当前 的 。 下 面 的 例子 演示 的 仅 仪 是 在 
Javal.0 版 的 状况 下 滚动 条 一 直 打 开 。 在 下 一 章 里 我 们 将 看 到 一 个 证 明 
Java 1.1 版 中 的 文字 区 域 的 例 程 。 


//: TextAreal.java 





























// Using the text area control 

import java.awt.*; 

import java.applet.*; 

public class TextAreai1 extends Applet { 
Button bi = new Button("Text Area 1"); 
Button b2 = new Button("Text Area 2"); 
Button b3 = new Button("Replace Text"); 
Button b4 = new Button("Insert Text"); 
TextArea ti = new TextArea("ti", 1, 30); 
TextArea t2 = new TextArea("t2", 4, 30); 
public void init() { 

add(b1); 


add(t1); 


} 


add(b2); 
add(t2); 
add(b3); 


add(b4); 


public boolean action (Event evt, Object arg) { 


else 


if(evt.target.equals(b1) ) 
getAppletContext().showStatus(t1.getText()); 

else if(evt.target.equals(b2)) { 
t2.setText("Inserted by Button 2"); 
t2.appendText(": " + ti.getText()); 
getAppletContext().showStatus(t2.getText()); 

} 

else if(evt.target.equals(b3)) { 
String s = " Replacement "; 
t2.replaceText(s, 3, 3 + s.length()); 

} 

else if(evt.target.equals(b4)) 
t2.insertText(" Inserted ", 10); 


// Let the base class handle it: 


return Super.action(evt, arg); 


return true; // We've handled it here 


t 
} ///:~ 





程序 中 有 几 个 不 同 的 “文本 区 域 ?构建 器 ， 这 其 中 的 一 个 在 此 处 显示 了 一 
个 初始 字符 串 和 行 号 和 列 号 。 不 同 的 按钮 显示 得 到 、 退 加 、 修 改 和 插入 





13.7 标签 


标签 准确 地 运作 : 安放 一 个 标签 到 窗 体 上 。 这 对 没有 标签 的 TextFields 

和 Text areas 来 说 非常 的 重要 ， 如 果 我 们 简单 地 想 安 放 文 字 的 信息 在 窗 
体 上 也 能 同样 的 使 用 。 我 们 能 像 本 章 中 第 一 个 例 程 中 演示 的 那样 ， 使 用 
drawStringO 里 边 的 paintO 在 确定 的 位 置 去 安置 一 个 文字 。 当 我 们 使 用 的 
标签 允许 我 们 通过 布局 管理 加 入 其 它 的 文字 组 件 。 “在 这 章 的 后 面 我 们 
将 进入 讨论 。) 


使 用 构建 器 我 们 能 创建 一 条 包括 初始 化 文字 的 标签 (这 是 我 们 典型 的 作 
法 ) ， 一 个 标签 包括 一 行 CENTER CREJ) 、LEFT ( 左 ) 和 
RIGHT( 右 ) 《静态 的 结果 取 整 定义 在 类 标签 里 ) 。 如 果 我 们 态 记 了 可 
以 用 getText() 和 getalignment() 读 取 值 ， 我 们 同样 可 以 用 setText() 和 
setAlignment() 来 改变 和 调整 。 下 面 的 例子 将 演示 标签 的 特点 : 


//: Labeli.java 








// Using labels 
import java.awt.*; 
import java.applet.*; 
public class Labeli extends Applet { 
TextField t1 = new TextField("ti", 10); 
Label labl1 = new Label("TextField t1"); 
Label labl2 = new Label(" an 
Label labl3 = new Label(" "y 
Label.RIGHT); 
Button b1 = new Button("Test 1"); 


Button b2 = new Button("Test 2"); 


public void init() { 
add(labl1); add(t1); 
add(b1); add(lab12); 
add(b2); add(lab13); 
} 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(b1)) 
labl2.setText("Text set into Label"); 
else if(evt.target.equals(b2)) { 
if(labl3.getText().trim().length() == 0) 
lab13.setText("lab13"); 
if(lab13.getAlignment() == Label.LEFT) 
lab13.setAlignment(Label.CENTER) ; 
else if(lab13.getAlignment()==Label.CENTER) 
lab13.setAlignment(Label.RIGHT); 
else if(lab13.getAlignment() == Label.RIGHT) 
lab13.setAlignment(Label.LEFT); 
} 
else 
return super.action(evt, arg); 


return true; 


} ///:~ 





首先 是 标签 的 最 典型 的 用 途 : 标记 一 个 文本 字段 或 文本 区 域 。 在 例 程 的 
第 二 部 分 ， 当 我 们 按 下 “test 1 按钮 通过 setTextO 将 一 串 空 的 空格 插入 到 
的 字段 里 。 因 为 空 的 空格 数 不 等 于 同样 的 字符 数 《〈 在 一 个 等 比例 间隔 的 
字库 里 ) ， 当 插入 文字 到 标签 里 时 我 们 会 看 到 文字 将 被 省 略 掉 。 在 例子 
的 第 三 部 分 保留 的 空 的 空格 在 我 们 第 一 次 按 下 “test 2? 会 发 现 标 签 是 空 的 
Gtrim0 删 除了 每 个 字符 串 结 尾部 分 的 空格 ) 并 且 在 开头 的 左 列 插入 了 
a 0 
I 效果 。 


我 们 可 能 会 认为 我 们 可 以 创建 一 个 空 的 标签 ， 然 后 用 setText0) 安 放 文 字 
在 里 面 。 然 而 我 们 不 能 在 一 个 空 标 签 内 加 入 文字 一 这 大 概 是 因为 空 标签 
没有 宽度 一 所 以 创建 一 个 没有 文字 的 空 标签 是 没有 用 处 的 。 在 上 面 的 例 
YH, “blank”" 标 俭 里 充满 空 的 空格 ， 所 以 它 足 够 容纳 后 面 加 入 的 文字 。 


同样 的 ，setAlignmentO 在 我 们 用 构建 器 创建 的 典型 的 文字 标签 上 没有 作 
用 。 这 个 标签 的 宽度 就 是 文字 的 宽度 ， 所 以 不 能 对 它 进行 任何 的 调整 。 

但 是 ， 如 果 我 们 启动 一 个 长 标签 ， 然 后 把 它 变 成 短 的 ， 我 们 就 可 以 看 到 
调整 的 效果 。 


这 些 导致 事件 连同 它们 最 小 化 的 矿 寸 被 挤 压 的 状况 被 程序 片 使 用 的 默认 
布局 管理 器 所 发 现 。 有 关 布 局 管理 器 的 部 分 包含 在 本 章 的 后 面 。 




















13.8 复 选 框 


复 选 框 提 供 一 个 制造 单一 选择 开关 的 方法 ， 它 包括 一 个 小 框 和 一 个 标 
答 。 典 型 的 复 选 框 有 一 个 小 的 “X”( 或 者 它 设 置 的 其 它 类 型 ) 或 是 空 
的 ， 这 依靠 项 目 是 否 被 选择 来 决定 的 。 


我 们 会 使 用 构建 右 正 党 地 创建 一 个 复 选 枉 ， 使 用 它 的 标签 来 充当 它 的 自 

变量 。 如 采 我 们 在 创建 复 选 框 后 想 读 出 或 改变 它 ， 我 们 能 够 获取 和 设置 

它 的 状态 ， 同 样 也 能 获取 和 设置 它 的 标签 。 注 意 ， 复 选 框 的 大 写 是 与 其 

它 的 控制 相 了 矛盾 的 。 

无 论 何 时 一 个 复 选 框 都 可 以 设置 和 清除 一 个 事件 指令 ， 我 们 可 以 捕捉 同 

Ts 
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//: CheckBox1.java 




















// Using check boxes 

import java.awt.*; 

import java.applet.*; 

public class CheckBox1 extends Applet { 
TextArea t = new TextArea(6, 20); 
Checkbox cb1 = new Checkbox("Check Box 1"); 
Checkbox cb2 = new Checkbox("Check Box 2"); 
Checkbox cb3 = new Checkbox("Check Box 3"); 
public void init() { 


add(t); add(cb1); add(cb2); add(cb3); 


public boolean action (Event evt, Object arg) { 
if(evt.target.equals(cb1) ) 
trace("1", cbi.getState()); 
else if(evt.target.equals(cb2) ) 
trace("2", cb2.getState()); 
else if(evt.target.equals(cb3) ) 
trace("3", cb3.getState()); 
else 
return super.action(evt, arg); 
return true; 
} 
void trace(String b, boolean state) { 
if(state) 
t.appendText("Box " + b + " Set\n"); 
else 
t.appendText("Box " + b + " Cleared\n"); 
} 
} ///:~ 





trace() 方 法 将 选中 的 复 选 框 名 和 当前 状态 用 appendTextO 发 送 到 文字 区 域 
中 去 ， 所 以 我 们 看 到 一 个 累积 的 被 选中 的 复 选 框 和 它们 的 状态 的 列表 。 


13.9 单 选 钮 


单 选 钮 在 GUI 程序 设计 中 的 概念 来 自 于 老式 的 电子 管 汽车 收音 机 的 机 械 
按钮 ， 当 我 们 按 下 一 个 按钮 和 时， 其 它 的 按钮 束 会 弹 起 。 因 此 它 允 许 我 们 
强制 从 众多 选择 中 作出 单一 选择 。 


AWT 没 有 单独 的 描述 单 选 钮 的 类 ; 取而代之 的 是 复 用 复 选 框 。 然 而 将 
复 选 框 放 在 单 选 钮 组 中 《并 且 修 改 它 的 外 形 使 它 看 起 来 不 同 于 一 般 的 复 
选 框 ) 我 们 必须 使 用 一 个 特殊 的 构建 器 象 一 个 自 变 量 一 样 的 作用 在 
checkboxGroup 对 象 上 。 “我 们 同样 能 在 创建 复 选 框 后 调用 
setCheckboxGroup0 方 法 。) 


一 个 复 选 框 组 没有 构建 融 的 目 变 量 ; 它 存 在 的 唯一 理由 就 是 聚集 一 些 复 
选 框 到 单 选 钮 组 里 。 一 个 复 选 框 对 象 必须 在 我 们 试图 显示 单 选 钮 组 之 前 
将 它 的 状态 设置 成 Iue， 否 则 在 运行 时 我 们 就 会 得 到 一 个 异常 。 如 果 我 
们 设置 超过 一 个 的 单 选 钮 为 tue， 只 有 最 后 的 一 个 能 被 设置 成 真 。 

这 里 有 个 简单 的 使 用 单 选 钮 的 例子 。 注 意 我 们 可 以 像 其 它 的 组 件 一 样 捕 
捉 单 选 钮 的 事件 : 


//: RadioButton1.java 

















// Using radio buttons 

import java.awt.*; 

import java.applet.*; 

public class RadioButtoni extends Applet { 
TextField t = 

new TextField("Radio button 2", 30); 

CheckboxGroup g = new CheckboxGroup(); 
Checkbox 


cb1 = new Checkbox("one", g, false), 


cb2 = new Checkbox("two", g, true), 
cb3 = new Checkbox("three", g, false); 
public void init() { 
t.setEditable(false); 
add(t); 
add(cb1); add(cb2); add(cb3); 
} 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(cb1)) 
t.setText("Radio button 1"); 
else if(evt.target.equals(cb2) ) 
t.setText("Radio button 2"); 
else if(evt.target.equals(cb3) ) 
t.setText("Radio button 3"); 
else 
return super.action(evt, arg); 
return true; 
} 
Po Pf 


显示 的 状态 是 一 个 文字 字段 在 被 使 用 。 这 个 字段 被 设置 为 不 可 编辑 的 ， 
因为 它 只 是 用 来 显示 数据 而 不 是 收集 。 这 演示 了 一 个 使 用 标签 的 可 取 之 
注意 字段 内 的 文字 是 由 最 早 选 择 的 单 选 钮 “Radio button 2” 初始 化 
Ji 











我 们 可 以 在 窗 体 中 拥有 相当 多 的 复 选 框 组 。 


13.10 下 拉 列 表 


下 拉 列 表 像 一 个 单 选 钮 组 ， 它 是 强制 用 户 从 一 组 可 实现 的 选择 中 选择 一 
个 对 象 的 方法 。 而 且 ， 它 是 一 个 实现 这 点 的 相当 简洁 的 方法 ， 也 最 易 改 
变 选 择 而 不 至 使 用 户 感到 吃力 (我 们 可 以 动态 地 改变 单 选 钮 ， 但 那 种 方 
法 显然 不 方便 ) 。Java 的 选择 框 不 像 windows 中 的 组 合 框 可 以 让 我 从 列 
表 中 选择 或 输入 目 己 的 选择 。 在 一 个 选择 框 中 你 只 能 从 列表 中 选择 仅仅 
一 个 项 目 。 在 下 面 的 例子 里 ， 选 择 框 从 一 个 确定 输入 的 数字 开始 ， 然 后 
当 按 下 一 个 按钮 时 ， 新 输入 的 数字 增加 到 框 里 。 你 将 可 以 看 到 选择 框 的 
一 些 有 趣 的 状态 : 


//: Choice1.java 





// Using drop-down lists 

import java.awt.*; 

import java.applet.*; 

public class Choice1 extends Applet { 

String[] description = { "Ebullient", "Obtuse", 
"Recalcitrant", "Brilliant", "Somnescent", 
"Timorous", "Florid", "Putrescent" }; 

TextField t = new TextField(30); 

Choice c = new Choice(); 

Button b = new Button("Add items"); 

int count = 0; 

public void init() { 
t.setEditable(false); 


for(int i = 0; i < 4; i++) 


c.addItem(description[count++]); 
add(t); 
add(c); 
add(b); 
} 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(c)) 
t.setText("index: " + c.getSelectedIndex() 
+" " + (String)arg); 
else if(evt.target.equals(b)) { 
if(count < description. length) 
c.addItem(description[count++]); 
} 


else 


return super.action(evt, arg); 


return true; 


} 


} ///:~ 


文本 字 字 段 中 显示 的 “selected index," 也 就 是 当前 选择 的 项 目的 序列 号 ， 


在 事件 中 选择 的 字符 串 就 像 action() 的 第 二 个 自 变量 的 字 串 


好 。 





Ar 


付 


描述 的 一 样 


运行 这 个 程序 片 时 ， 请 注意 对 Choice 框 大 小 的 判断 : 在 windows 里 ， 这 
个 大 小 是 在 我 们 拉 下 列表 时 确定 的 。 这 意味 着 如 果 我 们 拉 下 列表 ， 然 后 


增加 更 多 的 项 目 到 列表 中 ， 这 项 目 将 在 那 ， 但 这 个 下 拉 列 表 不 再 接受 

(我 们 可 以 通过 项 目 来 滚动 观察 一 注释 由 ) 。 然 而 ， 如 果 我 们 在 第 一 
次 拉 下 下 拉 列 表 前 将 所 的 项 目 竣 入 下 拉 列 表 ， 它 的 大 小 就 会 合适 。 当 

然 ， 用 户 在 使 用 时 和 希望 看 到 整个 的 列表 ， 所 以 会 在 下 拉 列 表 的 状态 里 对 
增加 项 目 到 选择 框 里 加 以 特殊 的 限定 。 


O: 这 一 行为 显然 是 一 种 错误 ， 会 Java 以 后 的 版 本 里 解决 。 





13.11 列表 框 


列表 框 与 选择 框 有 完全 的 不 同 ， 而 不 仅仅 是 当 我 们 在 激活 选择 框 时 的 显 
示 不 同 ， 列 表 框 固定 在 屏幕 的 指定 位 置 不 会 改变 。 男 外 ， 一 个 列表 框 允 
许多 个 选择 :如果 我 们 单 击 在 超过 一 个 的 项 目 上 ， 未 选择 的 则 表现 为 高 
亮度 ， 我 们 可 以 选择 象 我 们 想 要 的 一 样 的 多 。 如 果 我 们 想 察 看 项 目 列 
表 ， 我 们 可 以 调用 getSelectedItem() 来 产生 一 个 被 选择 的 项 目 列表 。 要 想 
从 一 个 组 里 删除 一 个 项 目 ， 我 们 必须 再 一 次 的 单 击 它 。 列 表 框 ， 当 然 这 
里 有 一 个 问题 束 是 它 默 认 的 动作 是 双击 而 不 是 单 击 。 单 击 从 组 中 增加 或 
删除 项 目 ， 双 击 调用 action0。 解 决 这 个 问题 的 方法 是 象 下 面 的 程序 假设 
的 一 样 重新 培训 我 们 的 用 户 。 


//: List1.java 











// Using lists with action() 
import java.awt.*; 
import java.applet.*; 
public class List1 extends Applet { 
String[] flavors = { "Chocolate", "Strawberry", 
"Vanilla Fudge Swirl", "Mint Chip", 
"Mocha Almond Fudge", "Rum Raisin", 
"Praline Cream", "Mud Pie" }; 
// Show 6 items, allow multiple selection: 
List lst = new List(6, true); 
TextArea t = new TextArea(flavors.length, 30); 
Button b = new Button("test"); 


int count = 0; 


public void init() { 
t.setEditable(false); 
for(int i = 0; i < 4; i++) 
1lst.addItem(flavors[count++]); 
add(t); 
add(lst); 
add(b); 
} 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(lst)) { 
t.setText(""); 
String[] items = lst.getSelectedItems(); 
for(int i = 0; i < items.length; i++) 
t.appendText(items[i] + "\n"); 
} 
else if(evt.target.equals(b)) { 
if(count < flavors.length) 
lst.addItem(flavors[count++], 0); 
} 
else 
return super.action(evt, arg); 


return true; 


} ///:~ 


按 下 按钮 时 ， 按 钮 增加 项 目 到 列表 的 顶部 〈 因 为 addItem0 的 第 二 个 目 变 
量 为 零 ) 。 增 加 项 目 到 列表 框 比 到 选择 框 更 加 的 合理 ， 因 为 用 户 期 望 去 
深 动 一 个 列表 框 〈 因 为 这 个 原因 ， 它 有 内 建 的 深 动 条 ) 但 用 户 并 不 愿意 
像 在 前 面 的 例子 里 不 得 不 去 计算 怎样 才能 滚动 到 要 要 的 那个 项 目 。 


然而 ， 调 用 action0) 的 唯一 方法 就 是 通过 双击 。 如 果 我 们 想 监 视 用 户 在 我 
0 rete ne a Me meen cele aa 
方法 。 











13.11.1 handleEvent() 


到 目前 为 止 ， 我 们 已 使 用 了 action0)， 现 有 另 一 种 方法 handleEventO 可 对 
每 一 事件 进行 尝试 。 当 一 个 事件 发 生 时 ， 它 总 是 针对 单独 事件 或 发 生 在 
单独 的 事件 对 象 上 。 该 对 象 的 handleEvent0) 方 法 是 自动 调用 的 ， 并 且 是 
被 handleEvent() 创 建 并 传递 到 handleEvent() 里 。 默 认 的 handleEvent() 
(handleEvent() 定 义 在 组 件 里 ， 基 础 类 的 所 有 控件 都 在 AWT 里 ) 将 像 我 
们 以 前 一 样 调用 action0 或 其 它 同 样 的 方法 去 指明 鼠标 的 活动 、 键 盘活 动 
或 者 指明 移动 的 焦点 。 我 们 将 会 在 本 章 的 后 面部 分 看 到 。 


如 果 其 它 的 方法 一 特别 是 action0 一 不 能 满足 我 们 的 需要 怎么 办 呢 ? 至 于 
列表 框 ， 例 如 ， 如 果 我 想 捕 捉 鼠 标 单 击 ， 但 action0 只 啊 应 双击 怎么 办 
We? 这 个 解答 是 过 载 handleEvent0， 毕 葛 它 是 从 程序 片 中 得 到 的 ， 因 此 
可 以 过 载 任何 非 确定 的 方法 。 当 我 们 为 程序 片 过载 handleEventO 时 ， 我 
们 会 得 到 所 有 的 事件 在 它们 发 送出 去 之 前 ， 所 以 我 们 不 能 假设 “这 里 有 
我 的 按钮 可 做 的 事件 ， 所 以 我 们 可 以 假设 按钮 被 按 下 了 ”从 它 被 action() 
设 为 真 值 。 在 handleEventO) 中 按钮 拥有 焦点 且 某 人 对 它 进行 分 配 都 是 可 
能 的 。 不 论 它 合理 与 否 ， 我 们 可 测试 这 些 事 件 并 遵照 handleEvent() 来 进 
行 操作 。 


为 了 修改 列表 样本 ， 使 它 会 啊 应 鼠标 的 单 击 ， 在 action0) 中 按钮 测试 将 被 
过 载 ， 但 代码 会 处 理 的 列表 将 像 下 面 的 例子 被 移 进 handleEvent() 中 去 : 


//: List2.java 




















// Using lists with handleEvent() 


import java.awt.*; 
import java.applet.*; 
public class List2 extends Applet { 

String[] flavors = { "Chocolate", "Strawberry", 
"Vanilla Fudge Swirl", "Mint Chip", 
"Mocha Almond Fudge", "Rum Raisin", 
"Praline Cream", "Mud Pie" }; 

// Show 6 items, allow multiple selection: 

List lst = new List(6, true); 

TextArea t = new TextArea(flavors.length, 30); 

Button b = new Button("test"); 

int count = 0; 

public void init() { 
t.setEditable(false); 
for(int i = 0; i < 4; i++) 

lst.addItem(flavors[count++]); 
add(t); 
add(lst); 
add(b); 

} 

public boolean handleEvent(Event evt) { 
if(evt.id == Event.LIST_SELECT | | 


evt.id == Event.LIST_DESELECT) { 


if(evt.target.equals(lst)) { 
t.setText(""); 
String[] items = lst.getSelectedItems(); 
for(int i = 0; i < items.length; i++) 
t.appendText(items[i] + "\n"); 
} 
else 
return super.handleEvent(evt); 
} 
else 
return super.handleEvent(evt); 
return true; 
} 
public boolean action(Event evt, Object arg) { 
if(evt.target.equals(b)) { 
if(count < flavors.length) 
lst.addItem(flavors[count++], 0); 
} 
else 
return super.action(evt, arg); 


return true; 


} ///:~ 


这 个 例子 同 前 面 的 例子 相同 除了 增加 了 handleEventO 外 简直 一 模 一 样 。 
在 程序 中 做 了 试验 来 验证 是 否 列表 框 的 选择 和 非 选择 存在 。 现 在 请 记 
住 ，handleEvent() 被 程序 片 所 过 载 ， 所 以 它 能 在 窗 体 中 任何 存在 ， 并 且 
被 其 它 的 列表 当成 事件 来 处 理 。 因 此 我 们 同样 必须 通过 试验 来 观察 目 
标 。【〔 昌 然 在 这 个 例子 中 ， 程 序 片 中 只 有 一 个 列表 框 所 以 我 们 能 假设 所 
有 的 列表 框 事件 必须 服务 于 列表 框 。 这 是 一 个 不 好 的 习惯 ,一旦 其 它 的 
列表 框 加 入 ， 它 就 会 变 成 程序 中 的 一 个 缺陷 。) 如 果 列 表 框 匹配 一 个 我 
们 感 兴趣 的 列表 框 ， 像 前 面 的 一 样 的 代码 将 按 上 面 的 策略 来 运行 。 注 意 
handleEvent() 的 窗 体 与 action() 的 相同 : 如 果 我 们 处 理 一 个 单独 的 事件 ， 
将 返回 真 值 ， 但 如 果 我 们 对 其 它 的 一 些 事件 不 感 兴趣 ， 通 过 
handleEvent0) 我 们 必须 返回 super.handleEventO 值 。 这 便 是 程序 的 核心 ， 
如 果 我 们 不 那样 做 ， 其 它 的 任何 一 个 事件 处 理 代 码 也 不 会 被 调用 。 例 
如 ， 试 注解 在 上 面 的 代码 中 返回 super.handleEvent(evt) 的 值 。 我 们 将 发 
现 action0 没 有 被 调用 ， 当 然 那 不 是 我 们 想得到 的 。 对 action0 和 
handlEventO 而 言 ， 最 重要 的 是 跟着 上 面 例子 中 的 格式 ， 并 且 当 我 们 目 
己 不 处 理事 件 时 一 直 返 回 基础 类 的 方法 版 本 信息 。《 在 例子 中 我 们 将 返 
HAA) o (幸运 的 是 ， 这 些 类 型 的 错误 的 仅 属于 Java ne 在 本 章 
后 面 将 看 到 的 新 设计 的 Java 1.1 消 除了 这 些 类 型 的 错误 。 


在 windows 里 ， 如 果 我 们 按 下 shift 键 ， 列 表 框 自动 允许 我 们 做 多 个 选 
择 。 这 非常 的 棒 ， 因 为 它 人 允许 用 户 做 单个 或 多 个 的 选择 而 不 是 编程 期 间 
固定 的 。 我 们 可 能 会 认为 我 们 变 得 更 加 的 精明 ， 并 且 当 一 个 鼠标 单 击 被 
evt.shiftdown() 产 生 时 如 果 shift 键 是 按 下 的 将 执行 我 们 自己 的 试验 程序 。 
AWT 的 设计 妨碍 了 我 们 一 我 们 不 得 不 去 了 解 哪 个 项 目 被 鼠标 点 击 时 是 
否 按 下 了 shift 键 ， 所 以 我 们 能 取消 其 余部 分 所 有 的 选择 并 且 只 选择 那 一 
个 。 不 管 怎样 ， 我 们 是 不 可 能 在 Java 1.0 版 中 做 出 来 的 。 (Java 1.1 将 所 
有 的 鼠标 、 键 盘 、 焦 点 事件 传送 到 列表 中 ， 所 以 我 们 能 够 完成 它 。) 




















13.12 布局 的 控制 


在 Java 里 该 方法 是 安 一 个 组 件 到 一 个 窗 体 中 去 ， 它 不 同 我 们 使 用 过 的 其 
它 GUI 系 统 。 首 先 ， 它 是 全 代码 的 ; 没有 控制 安放 组 件 的 “资源 ”。 其 
次 ， 该 方法 的 组 件 被 安放 到 一 个 被 “布局 管理 器 ”控制 的 窗 体 中 ， 由 “ 布 
局 管理 器 ”根据 我 们 add0 它 们 的 决定 来 安放 组 件 。 大 小 ， 形 状 ， 组 件 位 
置 与 其 它 系统 的 布局 管理 器 显著 的 不 同 。 另 外 ， 布 局 管理 器 使 我 们 的 程 
序 片 或 应 用 程序 适合 窗口 的 大 小 ， 所 以 ， 如 果 窗 口 的 尺寸 改变 (例如 ， 
ena eran ， 组 件 的 大 小 ， 形 状 和 位 置 都 会 改 


程序 片 和 帧 类 都 是 来 源 于 包含 和 显示 组 件 的 容器 。 (这 个 容 右 也 是 一 个 
组 件 ， 所 以 它 也 能 啊 应 事件 。) 在 容器 中 ， 调 用 setLayout() 方 法 允许 我 
选择 不 同 的 布局 管理 器 。 


在 这 节 里 我 们 将 探索 不 同 的 布局 管理 器 ， 并 安放 按钮 在 它们 之 上 。 这 里 
没有 捕捉 按钮 的 事件 ， 正 好 可 以 演示 如 何 布置 这 些 按钮 。 


13.12.1 FlowLayout 


到 目前 为 止 ， 所 有 的 程序 请 都 被 建立 ， 看 起 来 使 用 一 些 不 可 思议 的 内 部 
逻辑 来 布置 它们 的 组 件 。 那 是 因为 程序 使 用 一 个 默认 的 方式 : 
FlowLayout。 这 个 简单 的 “Flow” 的 组 件 安 装 在 窗 体 中 ， 从 左 到 右 ， 直 到 
项 部 的 空格 全 部 再 移 去 一 行 ， 并 继续 循环 这 些 组 件 。 


这 里 有 一 个 例子 明确 地 (当然 也 是 多 余地 ) 设置 一 个 程序 片 的 布局 管理 
髓 去 FlowLayout， 然 后 在 窗 体 中 安放 按钮 。 我 们 将 注意 到 FlowLayout 组 
件 使 用 它们 本 来 的 大 小 。 例 如 一 个 按钮 将 会 变 得 和 它 的 字 串 符 一 样 的 大 


小 。 


























//: FlowLayout1.java 


// Demonstrating the FlowLayout 
import java.awt.*; 


import java.applet.*; 


public class FlowLayout1 extends Applet { 
public void init() { 
setLayout(new FlowLayout()); 
for(int i = 0; i < 20; i++) 
add(new Button("Button " + i)); 
} 
} ///:~ 


所 有 组 件 将 在 FlowLayout 中 被 压缩 为 它们 的 最 小 玉 寸 ， 所 以 我 们 可 能 会 
得 到 一 些 奇 怪 的 状态 。 例 如 ， 一 个 标签 会 合适 它 目 己 的 字符 串 的 矿 才 ， 
所 以 它 会 右 对 齐 产生 一 个 不 变 的 显示 。 


13.12.2 BorderLayout 


布局 管理 器 有 四 边 和 中 间 区 域 的 概念 。 当 我 们 增加 一 些 事 物 到 使 用 
BorderLayout 的 面板 上 时 我 们 必须 使 用 add0 方 法 将 一 个 字符 串 对 象 作 为 
它 的 第 一 个 自 变 量 ， 并 且 字 符 串 必须 指定 (正确 的 大 

©) “North” CE) , “South”? CF) , “west” (Æ) ，“East”( 右 ) 或 
者 “Center”"。 如 果 我 们 拼写 错误 或 没有 大 写 ， 束 会 得 到 一 个 编译 时 的 错 
误 ， 并 且 程 序 片 不 会 像 你 所 期 望 的 那样 运行 。 笠 运 的 是 ， 我 们 会 很 快 发 
现在 Java 1.1 中 有 了 更 多 改进 。 


这 是 一 个 简单 的 程序 例子 : 


//: BorderLayout1.java 





// Demonstrating the BorderLayout 
import java.awt.*; 
import java.applet.*; 


public class BorderLayout1 extends Applet { 


public void init() { 
int i = 0; 
setLayout(new BorderLayout()); 
add("North", new Button("Button " + i++)); 
add("South", new Button("Button " + i++)); 
add("East", new Button("Button " + i++)); 
add("West", new Button("Button " + it+)); 
add("Center", new Button("Button " + i++)); 

} 

} ///:~ 


除了 “Center” 的 每 一 个 位 置 ， 当 元 素 在 其 它 空 间 内 扩大 到 最 大 时 ， 我 们 
会 把 它 压 缩 到 适合 空间 的 最 小 尺寸 。 但 是 ,，“Center” 扩 大 后 只 会 占据 中 
心 位 置 。 

BorderLayout 是 应 用 程序 和 对 话 框 的 默认 布局 管理 器 。 

13.12.3 GridLayout 


GridLayout 人 允许 我 们 建立 一 个 组 件 表 。 添 加 那些 组 件 时 ， 它 们 会 按 从 无 
到 右 、 从 上 到 下 的 顺序 在 网 格 中 排列 。 在 构建 器 里 ， 需 要 指定 自己 希望 
的 行 、 列 数 ， 它 们 将 按 正 比例 展开 。 


//: GridLayout1.java 


// Demonstrating the GridLayout 
import java.awt.*; 
import java.applet.*; 


public class GridLayout1 extends Applet { 


public void init() { 
setLayout(new GridLayout(7,3)); 
for(int i = 0; i < 20; i++) 
add(new Button("Button " + i)); 
} 
} ///:~ 


在 这 个 例子 里 共有 21 个 空位 ， 但 却 只 有 20 个 按钮 ， 最 后 的 一 个 位 置 作 留 
空 处 理 ; 注意 对 GridLayout 来 说 ， 并 不 存在 什么 “均衡 ”处理 。 


13.12.4 CardLayout 


CardLayout 人 允许 我 们 在 更 复杂 的 拥有 真正 的 文件 夹 卡 片 与 一 条 边 相 过 的 
环境 里 创建 大 致 相同 于 “卡片 式 对 话 框 ?的 布局 ， 我 们 必须 压 下 一 个 卡 所 
使 不 同 的 对 话 框 带 到 前 面 来 。 在 AWT 里 不 是 这 样 的 : CardLayout 是 简单 
的 空 的 空格 ， 我 们 可 以 自由 地 把 新 卡片 带 到 前 面 来 。 (JFC/Swing 库 包 
括 卡 片 式 的 窗 格 看 起 来 非常 的 棒 ， 且 可 以 我 们 处 理 所 有 的 细节 。 ) 


1. 联合 布局 (Combining layouts) 


下 面 的 例子 联合 了 更 多 的 布局 类 型 ， 在 最 初 只 有 一 个 布局 管理 需 被 程序 
片 或 应 用 程序 操作 看 起 来 相当 的 困难 。 这 是 事实 ， 但 如 采 我 们 创建 更 多 
的 面板 对 象 ， 每 个 面板 都 能 拥有 一 个 布局 管理 器 ， 并 且 像 被 集成 到 程序 
片 或 应 用 程序 中 一 样 使 用 程序 片 或 应 用 程序 的 布局 管理 器 。 这 就 家 下 面 
程序 中 的 一 样 给 了 我 们 更 多 的 灵活 性 : 


//: CardLayout1.java 








// Demonstrating the CardLayout 
import java.awt.*; 


import java.applet.Applet; 


class ButtonPanel extends Panel { 
ButtonPanel(String id) { 
setLayout(new BorderLayout()); 


add("Center", new Button(id)); 


} 


public class CardLayout1 extends Applet { 
Button 
first = new Button("First"), 
second = new Button("Second"), 
third = new Button("Third"); 
Panel cards = new Panel(); 
CardLayout cl = new CardLayout(); 
public void init() { 
setLayout(new BorderLayout()); 
Panel p = new Panel(); 
p.setLayout(new FlowLayout()); 
p.add(first); 
p.add(second); 
p.add(third); 
add("North", p); 
cards.setLayout(cl); 


cards.add("First card", 


new ButtonPanel("The first one")); 
cards.add("Second card", 
new ButtonPanel("The second one") ); 
cards.add("Third card", 
new ButtonPanel("The third one")); 
add("Center", cards); 
} 
public boolean action(Event evt, Object arg) { 
if (evt.target.equals(first)) { 


cl.first(cards); 


else if (evt.target.equals(second)) { 
cl.first(cards); 


cl.next(cards); 


else if (evt.target.equals(third)) { 


cl.last(cards); 


else 


return Super.action(evt, arg); 


return true; 


de 


这 个 例子 首先 会 创建 一 种 新 类 型 的 面板 : BottonPanel (按钮 面板 ) . E 
包括 一 个 单独 的 按钮 ， 安 放 在 BorderLayout 的 中 央 ， 那 意味 着 它 将 充满 
整个 的 面板 。 按 钮 上 的 标签 将 让 我 们 知道 我 们 在 CardLayout 上 的 那个 面 
板 上 。 





在 程序 片 里 ， 面 板 卡片 上 将 存放 卡片 和 布局 管理 器 CL 因 为 CardLayout 必 
须 组 成 类 ， 因 为 当 我 们 需要 处 理 卡 片 时 我 们 需要 访问 这 些 句柄 。 


这 个 程序 请 变 成 使 用 BorderLayonut 来 取代 它 的 默认 FlowLayout， 创 建 面 

板 来 容纳 三 个 按钮 〈 使 用 FlowLayout) ， 并 且 这 个 面板 安置 在 程序 片 末 
卡片 面板 增加 到 程序 片 的 “Center" 里 ， 有 效 地 占据 面板 的 
其 余地 方 。 


当 我 们 增加 BottonPanels( 或 者 任何 其 它 我 们 想 要 的 组 件 ) 到 卡片 面板 
时 ，add() 方 法 的 第 一 个 自 变 量 不 是 “North”, “South” 等 等 。 相 反 的 是 ， 
它 是 一 个 描述 卡 卢 的 字符 串 。 如 果 我 们 想 轻 击 那 张 卡片 使 用 字符 串 ， 我 
们 就 可 以 使 用 ， 虽 然 这 字符 串 不 会 显示 在 卡片 的 任何 地 方 。 使 用 的 方法 
不 是 使 用 action(); 代 之 使 用 first0、nextO0 和 1lastO 等 方法 。 请 查看 我 们 有 
关 其 它 方 法 的 文件 。 


在 Java 中 ， 使 用 的 一 些 卡 片 式 面板 结构 十 分 的 重要 ， 因 为 (我 们 将 在 后 
面 看 到 〉 在 程序 片 编 程 中 使 用 的 弹出 式 对话 框 是 十 分 令 人 泪 表 的 。 对 于 
J o ea CardLayout 是 唯一 有 效 的 取得 很 多 不 同 的 “ 弹 
出 式 ” 的 窗 体 。 


13.12.5 GridBagLayout 


很 早 以 前 ， 人 们 相信 所 有 的 恒星 、 行 星 、 太 阳 及 月 亮 都 围绕 地 球 公转 。 
这 是 直观 的 观察 。 但 后 来 天 文学 家 变 得 更 加 的 精明 ， 他 们 开始 跟踪 个 别 
星体 的 移动 ， 它 们 中 的 一 些 似乎 有 时 在 轨道 上 缓慢 运行 。 因 为 天 文学 家 
知道 所 有 的 天 体 都 围绕 地 球 公转 ， 天 文学 家 花费 了 大 量 的 时 间 来 讨论 相 
关 的 方程 式 和 理论 去 解释 天 体 对 象 的 运行 。 当 我 们 试图 用 
GridBagLayout 来 工作 时 ， 我 们 可 以 想像 自己 为 一 个 早期 的 天 文学 家 。 
基础 的 条 例 是 (公告 ， 有趣 的 是 设计 者 居然 在 太阳 上 (这 可 能 是 在 天 体 
图 中 标 错 了 位 置 所 致 ， 译 者 注 )) 所 有 的 天 体 都 将 遵守 规则 来 运行 。 哥 
白 尼 日 新 说 《又 一 次 不 顾 嘲 讽 ， 发 现 太 阳 系 内 的 所 有 的 行星 围绕 太阳 公 



































转 。) 是 使 用 网 络 图 来 判断 布局 ， 这 种 方法 使 得 程序 员 的 工作 变 得 简 
单 。 直 到 这 些 增加 到 Java 里 ， 我 们 忍耐 〈 持 续 的 冷 哺 热 讽 ) 西班牙 的 
GridBagLayout 和 GridBagConstraints 狂 热 宗 教 。 我 们 建议 废止 
GridBagLayout。 取 代 它 的 是 ， 使 用 其 它 的 布局 管理 器 和 特殊 的 在 单个 
程序 里 联合 几 个 面板 使 用 不 同 的 布局 管理 器 的 技术 。 我 们 的 程序 片 看 起 
来 不 会 有 什么 不 同 ; 至 少 不 足 以 调整 GridBagLayout 限 制 的 麻烦 。 对 我 
而 言 ， 通 过 一 个 例子 来 讨论 它 实 在 是 令 人 头痛 (并 且 我 不 鼓励 这 种 库 设 
th) 。 相 反 ， 我 建议 您 从 阅读 Cornell/ 和 Horstmann 撰 写 的 《核心 Java》 
《第 二 版 ，Prentice-Hall 出 版 社 ，1997 年 ) 开始 。 


在 这 范围 内 还 有 其 它 的 : 在 下 C/Swing 库 里 有 一 个 新 的 使 用 Smalltalk 的 
受 人 欢迎 的 “Spring and Struts” 布 局 管理 器 并 且 它 能 显著 地 减少 
GridBagLayout 的 需要 。 








13.13 action 的 蔡 代 品 


正如 早先 指出 的 那样 ，action0 并 不 是 我 们 对 所 有 事 进行 分 类 后 自动 为 

handleEventO 调 用 的 唯一 方法 。 有 三 个 其 它 的 被 调用 的 方法 集 ， 如 果 我 
们 想 捕 捉 某 些 类 型 的 事件 (键盘 、 鼠 标 和 焦点 事件 ) ， 因 此 我 们 不 得 不 
过 载 规定 的 方法 。 这 些 方法 是 定义 在 基础 类 组 件 里 ， 所 以 他 们 几乎 在 所 
有 我 们 可 能 安放 在 窗 体 中 的 组 件 中 都 是 有 用 的 。 然 而 ， 我 们 也 注意 到 这 
种 方法 在 Java ”1.1 版 中 是 不 被 文 持 的 ， 同 样 尽 管 我 们 可 能 注意 到 继承 代 
码 利 用 了 这 种 方法 ， 我 们 将 会 使 用 Java 1.1 版 的 方法 来 代 检 (本 章 后 面 
有 详细 介绍 ) 。 


组 件 方法 何 时 调用 


action(Event evt, Object what) 当 典 型 的 事件 针对 组 件 发 生 《〈 例 如 ， 当 按 
下 一 个 按钮 或 下 拉 列 表 项 目 被 选中 ) 时 调用 


keyDown(Event evt, int key) 当 按 键 被 按 下 ， 组 件 拥有 焦点 时 调用 。 第 
二 个 目 变 量 是 按 下 的 键 并 且 是 元 余 的 是 从 evtkey 处 复制 来 的 


keyup(Event evt, int key) 当 按 键 被 释放 ， 组 件 拥有 焦点 时 调用 


lostFocus(Event evt, Object what) 焦点 从 目标 处 移 开 时 调用 。 通 常 ，what 
是 从 evt.arg 里 见 余 复制 的 


gotFocus(Event evt, Object what) 焦点 移动 到 目标 时 调用 


























mouseDown(Event evt, int x, int y) 一 个 鼠标 按 下 存在 于 组 件 之 上 ,在 
X，Y 座 标 处 时 调用 


mouseUp(Event evt, int x, int y) 一 个 鼠标 升 起 存在 于 组 件 之 上 时 调用 
mouseMove(Event evt, int x, int y) 当 鼠 标 在 组 件 上 移动 时 调用 
mouseDrag(Event evt, int x, int y) 鼠标 在 一 次 mouseDown 事 件 发 生 后 拖 


动 。 所 有 拖 动 事件 都 会 报告 给 内 部 发 生 了 mouseDown 事 件 的 那个 组 件 ， 
直到 过 到 一 次 mouseUp 为 止 





mouseEnter(Event evt, int x, int y) 鼠标 从 前 不 在 组 件 上 方 ， 但 目前 在 
mouseExit(Event evt, int x, int y) 鼠标 曾经 位 于 组 件 上 方 ， 但 目前 不 在 


当 我 们 处 理 特殊 情况 时 一 一 一 个 鼠标 事件 ， 例 如 ， 它 恰好 是 我 们 想得到 
的 鼠标 事件 存在 的 座 标 ， 我 们 将 看 到 每 个 程序 接收 一 个 事件 连同 一 些 我 
们 所 需要 的 信息 。 有 趣 的 是 ， 当 组 件 的 handleEventO 调 用 这 些 方法 时 
(典型 的 事例 ) ， 附 加 的 自 变 量 总 是 多 余 的 因为 它们 包含 在 事件 对 象 
里 。 事 实 上 ， 如 果 我 们 观察 component.handleEvent() 的 源 代码 ， 我 们 能 
发 现 它 显然 将 增加 的 自 变 量 抽 出 事件 对 象 〈 这 可 能 是 考虑 到 在 一 些 语言 
中 无 效率 的 编码 ， 但 请 记 住 Java 的 焦点 是 安全 的 ， 不 必 担 心 。) 试验 对 
我 们 表明 这 些 事件 事实 上 在 被 调用 并 且 作为 一 个 有 趣 的 答 试 是 值得 创建 
一 个 过 载 每 个 方法 的 程序 片 ，〈action0 的 过 载 在 本 章 的 其 它 地方 ) 当 事 
件 发 生 时 显示 它们 的 相关 数据 。 


这 个 例子 同样 向 我 们 展示 了 怎样 制造 自己 的 按钮 对 象 ， 因 为 它 是 作为 目 
标的 所 有 事件 权益 来 使 用 。 我 可 能 会 首先 (也 是 必须 的 ) 假设 制造 一 个 
新 的 按钮 ， 我 们 从 按钮 处 继承 。 但 它 并 不 能 运行 。 取 而 代 之 的 是 ， 我 们 
从 画布 组 件 处 (一 个 非常 普通 组 件 ) 继承 ， 并 在 其 上 不 使 用 paint() 方 法 
画 出 一 个 按钮 。 正 如 我 们 所 看 到 的 ， 自 从 一 些 代码 混入 到 画 按 钮 中 去 ， 
按钮 根本 就 不 运行 ， 这 实在 是 太 粳 糕 了 。 “如 果 您 不 相信 我 ， 试 图 在 例 
子 中 为 画布 组 件 交 换 按钮 ， 请 记 住 调用 称 为 super 的 基础 类 构建 疾 。 我 们 
会 看 到 按钮 不 会 被 画 出 ， 事 件 也 不 会 被 处 理 。) 


myButton 类 是 明确 说 明 的 : 它 只 和 一 个 自动 事件 (AutoEvent)“ 父 窗 
口 ” 一 起 运行 〈 父 窗口 不 是 一 个 基础 类 ， 它 是 按钮 创建 和 存在 的 窗 

口 。) 。 通 过 这 个 知识 ，myButton 可 能 进入 到 父 窗口 并 且 处 理 它 的 文字 
字段 ， 必 然 就 能 将 状态 信息 写 入 到 父 窗口 的 字段 里 。 当 然 这 是 一 种 非常 
有 限 的 解决 方法 ，myButton 仅 能 在 连结 AutoEvent 时 被 使 用 。 这 种 代码 
有 时 称 为 “高 度 结合 ”。 但 是 ， 制 造 myButton 更 需要 很 多 的 不 是 为 例子 
(和 可 能 为 我 们 将 写 的 一 些 程 序 片 ) 担保 的 努力 。 再 者 ， 请 注意 下 面 的 
代码 使 用 了 Java 1.1 版 不 支持 的 API。 


//: AutoEvent.java 
































// Alternatives to action() 


import java.awt.*; 


import java.applet.*; 
import java.util.*; 
Class MyButton extends Canvas { 
AutoEvent parent; 
Color color; 
String label; 
MyButton(AutoEvent parent, 
Color color, String label) { 
this.label = label; 
this.parent = parent; 
this.color = color; 
} 
public void paint(Graphics g) { 
g.setColor (color); 
int rnd = 30; 
g.fillRoundRect(0, ©, size().width, 
size().height, rnd, rnd); 
g.setColor(Color.black) ; 
g.drawRoundRect(0, ©, size().width, 
size().height, rnd, rnd); 
FontMetrics fm = g.getFontMetrics(); 
int width = fm.stringWidth(label); 


int height = fm.getHeight(); 


int ascent = fm.getAscent(); 
int leading = fm.getLeading(); 
int horizMargin = (size().width - width)/2; 
int verMargin = (size().height - height)/2; 
g.setColor(Color.white) ; 
g.drawString(label, horizMargin, 
verMargin + ascent + leading); 

} 

public boolean keyDown(Event evt, int key) { 
TextField t = 

(TextField)parent.h.get("keyDown"); 

t.setText(evt.toString()); 
return true; 

} 

public boolean keyUp(Event evt, int key) { 
TextField t = 

(TextField)parent.h.get("keyUp"); 

t.setText(evt.toString()); 
return true; 

} 

public boolean lostFocus(Event evt, Object w) 
TextField t = 


(TextField)parent.h.get("lostFocus"); 


t.setText(evt.toString()); 
return true; 
} 
public boolean gotFocus(Event evt, Object w) 
TextField t = 
(TextField)parent.h.get("gotFocus"); 
t.setText(evt.toString()); 
return true; 
} 
public boolean 
mouseDown(Event evt,int x,int y) { 
TextField t = 
(TextField)parent.h.get("mouseDown" ) ; 
t.setText(evt.toString()); 
return true; 
} 
public boolean 
mouseDrag(Event evt,int x,int y) { 
TextField t = 
(TextField)parent.h.get("mouseDrag"); 
t.setText(evt.toString()); 


return true; 


public boolean 
mouseEnter(Event evt,int x,int y) { 
TextField t = 
(TextField)parent.h.get("mouseEnter") ; 
t.setText(evt.toString()); 
return true; 
} 
public boolean 
mouseExit(Event evt,int x,int y) { 
TextField t = 
(TextField)parent.h.get("mouseExit"); 
t.setText(evt.toString()); 
return true; 
} 
public boolean 
mouseMove(Event evt,int x,int y) { 
TextField t = 
(TextField)parent.h.get("mouseMove") ; 
t.setText(evt.toString()); 
return true; 
} 
public boolean mouseUp(Event evt,int x,int y) 


TextField t = 


(TextField)parent.h.get("mouseUp"); 
t.setText(evt.toString()); 


return true; 


} 


public class AutoEvent extends Applet { 

Hashtable h = new Hashtable(); 

String[] event = { 
"keyDown", "keyUp", "lostFocus", 
"gotFocus", "mouseDown", "mouseUp", 
"mouseMove", "mouseDrag", "mouseEnter", 
"mouseExit" 

}; 

MyButton 
b1 = new MyButton(this, Color.blue, "test1"), 


b2 


new MyButton(this, Color.red, "test2"); 
public void init() { 
setLayout(new GridLayout(event.length+1,2)); 
for(int i = 0; i < event.length; i++) { 
TextField t = new TextField(); 
t.setEditable(false); 
add(new Label(event[i], Label.CENTER) ); 


add(t); 


h.put(event[i], t); 
} 
add(b1); 
add(b2); 
} 
} ///:~ 





我 们 可 以 看 到 构建 器 使 用 利用 自 变 量 同名 的 方法 ， 所 以 目 变 量 被 赋值 ， 
并 且 使 用 this 来 区 分 : 


this.label = label; 


paint() 方 法 由 简单 的 开始 : 它 用 按钮 的 颜色 填充 了 一 个 “ 圆 角 矩形”， 然 
后 画 了 一 个 黑 线 围绕 它 。 请 注意 size0 的 使 用 决定 了 组 件 的 宽度 和 长 度 
(当然 ， 是 像素 ) 。 这 之 后 ，paintO 看 起 来 非常 的 复杂 ， 因 为 有 大 量 的 
预测 去 计算 出 怎样 利用 “font metrics” 集 中 按钮 的 标签 到 按钮 里 。 我 们 能 
得 到 一 个 相当 好 的 关于 继续 关注 方法 调用 的 主意 ， 它 将 程序 中 那些 相当 
平凡 的 代码 挑 出 ， 当 我 们 想 集中 一 个 标签 到 一 些 组 件 里 时 ， 我 们 正好 可 
DATE EAT BI AAI 


您 直到 注意 到 AutoEvent 类 才能 正确 地 理解 keyDown(),keyUp(0) 及 其 它 方 
法 的 运行 。 这 包含 一 个 Hashtable( 译 者 注 : 散 列 表 )〉 去 控制 字符 串 来 描 
述 关 于 事件 处 理 的 事件 和 TextField 类 型 。 当 然 ， 这 些 能 被 静态 的 创建 而 
不 是 放 入 Hashtable 但 我 认为 您 会 同意 它 是 更 容易 使 用 和 改变 的 。 特 别 
是 ， 如 果 我 们 需要 在 AutoEvent 中 增加 或 删除 一 个 新 的 事件 类 型 ， 我 们 
只 需要 简单 地 在 事件 列队 中 增加 或 删除 一 个 字符 串 一 一 所 有 的 工作 都 自 
动 地 完成 了 。 


我 们 碍 出 在 keyDown(0)，keyupO 及 其 它 方 法 中 的 字符 串 的 位 置 回 到 

myButton 中 。 这 些 方法 中 的 任何 一 个 都 用 父 句 柄 试图 回 到 父 窗口 。 父 类 
是 一 个 AutoEvent， 它 包含 Hashtable h 和 get0) 方 法 ， 当 拥有 特定 的 字符 串 
时 ， 将 对 一 个 我 们 知道 的 TextField 对 象 产 生 一 个 句柄 “〈 因 此 它 被 选派 到 
那 )。 然 后 事件 对 象 修改 显示 在 TextField 中 的 字符 串 陈述 。 从 我 们 可 以 
真正 注意 到 举 出 的 例子 在 我 们 的 程序 中 运行 事件 时 以 来 ， 可 以 发 现 这 个 




















例子 运行 起 来 占 为 有 趣 的 。 


13.14 程序 上 厂 的 局 限 


出 于 安全 缘故 ， 程 序 片 十 分 受到 限制 ， 并 且 有 很 多 的 事 我 们 都 不 能 做 。 
您 一 般 会 问 : 程序 片 看 起 来 能 做 什么 ， 传 闻 它 又 能 做 什么 : RED i as 
中 WEB 页 的 功能 。 自 从 作为 一 个 网 上 冲浪 者 ， 我 们 从 未 真正 想 了 解 是 
个 一 个 WEB 页 来 目 友 好 的 或 者 不 友好 的 站 点 ， 我 们 想 要 一 些 可 以 安全 
地 行动 的 代码 。 所 以 我 们 可 能 会 注意 到 大 量 的 限制 |: 


(1) 一 个 程序 片 不 能 接触 到 本 地 的 磁盘 。 这 意味 着 不 能 在 本 地 磁盘 上 写 
和 读 ， 我 们 不 想 一 个 程序 片 通 过 WEB 页 面 阅读 和 传送 重要 的 信息 。 写 
是 被 蔡 止 的 ， 当 然 ， 因 为 那 将 会 引起 病毒 的 侵入 。 妆 数字 签名 生效 时 ， 
这 些 限制 会 被 解除 。 


(2) 程序 片 不 能 拥有 沫 单 。〈 注 意 : 这 是 规定 在 Swing 中 的 ) 这 可 能 会 减 
少 关 于 安全 和 关于 程序 简化 的 麻烦 。 我 们 可 能 会 接 到 有 关 程 序 片 协调 利 
葵 以 作为 WEB 页 面 的 一 部 分 的 通知 ;而 我 们 通 种 不 去 注意 程序 刻 的 范 
围 。 这 儿 没 有 帧 和 标题 条 从 沫 单 处 弹出 ， 出 现 的 帧 和 标题 条 是 属于 
WEB 浏 览 器 的 。 也 许 将 来 设计 能 被 改变 成 允许 我 们 将 浏览 器 沫 单 和 程 
序 片 菜单 相 结合 起 来 一 一 程序 片 可 以 影响 它 的 环境 将 导致 太 危及 整个 系 
统 的 安全 并 使 程序 片 过 于 的 复 森 。 


(3) 对 话 框 是 不 被 信任 的 。 在 Java 中 ， 对 话 框 存在 一 些 令 人 难 解 的 地 方 。 
首先 ， 它 们 不 能 正确 地 拒绝 程序 片 ， 这 实在 是 令 人 泪 丧 。 如 果 我 们 从 程 
序 上 请 弹出 一 个 对 话 框 ， 我 们 会 在 对 话 框 上 看 到 一 个 附 上 的 消息 框 “不 被 
信任 的 程序 片 ”。 这 是 因为 在 理论 上 ， 它 有 可 能 欺骗 用 户 去 考虑 他 们 在 
通过 WEB 同 一 个 老 顾 客 的 本 地 应 用 程序 交易 并 且 让 他 们 输入 他 们 的 信 
用 卡号 。 在 看 到 AWT 开 发 的 那 种 GUI 后 ， 我 们 可 能 会 难过 地 相信 任何 人 
都 会 被 那 种 方法 所 患 弄 。 但 程序 片 是 一 直 附 着 在 一 个 Web 页 面 上 的 ， 并 
可 以 在 浏览 器 中 看 到 ， 而 对 话 框 没 有 这 种 依附 关系 ， 所 以 理论 上 是 可 能 
的 。 因 此 ， 我 们 很 少 会 见 到 一 个 使 用 对 话 框 的 程序 片 。 


在 较 新 的 浏览 器 中 ， 对 受到 信任 的 程序 厂 来 说 ， 许 多 限制 都 被 放 冤 了 
受信 任 程序 片 由 一 个 信任 源 认 证 ) 。 


涉及 程序 片 的 开发 时 ， 还 有 另 一 些 问 题 需要 考虑 : 















































国 程 序 片 不 停 地 从 一 个 适合 不 同类 的 单独 的 服务 器 上 下 载 。 我 们 的 浏览 
器 能 够 缓存 程序 片 ， 但 这 没有 保证 。 在 Java 1.1 版 中 的 一 个 改进 是 
JAR (Java ARchive) 文件 ， 它 允许 将 所 有 的 程序 片 组 件 (包括 其 它 的 
类 文件 、 图 像 、 声 音 ) 一 起 打包 到 一 个 的 能 被 单个 服务 器 处 理 下 载 的 压 
0 T 
文件 。 


四 因为 安全 方面 的 缘故 ， 我 们 做 某 些 工作 更 加 困难 ， 例 如 访问 数据 库 和 
发 送 电 子 邮 件 。 男 外 ， 安 全 限制 规则 使 访问 多 个 主机 变 得 非 第 的 困难 ， 
因为 每 一 件 事 都 必须 通过 WEB 服 务 器 路 由 ， 形 成 一 个 性 能 瓶颈 ， 并 且 
单一 环 市 的 出 错 都 会 导致 整个 处 理 的 停止 。 


轿 浏 多 如 里 的 程序 片 不 会 拥有 同样 的 本 地 应 用 程序 运行 的 控件 类 型 。 例 
如 ， 自 从 用 户 可 以 开关 页 面 以 来 ， 在 程序 片 中 不 会 拥有 一 个 形式 上 的 对 
话 框 。 当 用 户 对 一 个 WEB 页 面 进行 改变 或 退出 浏览 右 时 ， 对 我 们 的 程 
序 片 而 言 简 直 是 一 场 灾难 一 一 这 时 没有 办 法 保存 状态 ， 所 以 如 果 我 们 在 
处 理 和 操作 中 时 ， 信 息 会 被 丢失 。 另 外 ， 当 我 们 离开 一 个 WEB 页 面 
人 
\ 确 定 的 。 


13.14.1 程序 片 的 优点 


如 果 能 容忍 那些 限制 ， 那 么 程序 片 的 一 些 优点 也 是 非常 突出 的 ， 尤 其 是 
在 我 们 构建 客户 / 服务 器 应 用 或 者 其 它 网 络 应 用 时 : 


四 没有 安装 方面 的 和 争议。 程序 片 拥 有 真正 的 平台 独立 性 〈 包 括 容 易 地 播 
放声 音 文件 等 能 力 ) 所 以 我 们 不 需要 针对 不 同 的 平台 修改 代码 也 不 需要 
任何 人 根据 安装 运行 任何 的 “tweaking”。 事 实 上 ， 安 装 每 次 自动 地 将 

WEB 页 连同 程序 片 一 起 ， 因 此 安静 、 自 动 地 更 新 。 在 传统 的 客户 机 / 服 
务 嚣 系统 中 ， 建 并 和 安装 一 个 新 版 本 的 客户 端 软件 简直 就 是 一 场 恶 梦 。 


BA 为 安全 的 原因 创建 在 核心 Java 语 言 和 程序 片 结构 中 ， 我 们 不 必 担 心 
坏 的 代码 而 导致 毁坏 某 人 的 系统 。 这 样 ， 连 同 前 面 的 优点 ， 可 使 用 
Java〈 可 从 JavaScript 和 VBScript 中 选择 客户 端的 WEB 编 程 工具 ) 为 所 谓 
的 Intrant〈 在 公司 内 部 使 用 而 不 同 Pnternet 转 移 的 企业 内 部 网 络 ) 客户 机 / 
服务 器 开发 应 用 程序 。 


四 由 于 程序 片 是 目 动 同 HTML 集 成 的 ， 所 以 我 们 有 一 个 内 建 的 独立 平台 















































文件 系统 去 文 持 程 序 片 。 这 是 一 个 很 有 趣 的 方法 ， 因 为 我 们 惯 于 拥有 程 
序 文件 的 一 部 分 而 不 是 相反 的 拥有 文件 系统 。 


13.15 视窗 化 应 用 


出 于 安全 的 缘故 ， 我 们 会 看 到 在 程序 片 我 们 的 行为 非常 的 受到 限制 。 我 
们 真实 地 感到 ， 程 序 记 是 被 临时 地 加 入 在 WEB 浏 览 器 中 的 ， 因 此 ， 它 
的 功能 连同 它 的 相关 知识 ， 控 件 都 必须 加 以 限制 。 但 是 ， 我 们 希望 Java 
能 制造 一 个 开 窗 口 的 程序 去 运行 一 些 事 物 ， 否 则 宁愿 安放 在 一 个 WEB 
页 面 上 ， 并 且 也 许 我 们 希望 它 可 以 运行 一 些 可 靠 的 应 用 程序 ， 以 及 夸张 
的 实时 便携 性 。 在 这 本 书 前 面 的 章节 中 我 们 制造 了 一 些 命令 行 应 用 程 
序 ， 但 在 一 些 操作 环境 中 《例如 : Macintosh) 没有 命令 行 。 所 以 我 们 
有 很 多 的 理由 去 利用 Java 创 建 一 个 设置 窗口 ， 非 程序 片 的 程序 。 这 当然 


是 一 个 十 分 合理 的 要 求 。 


一 个 Java 设 置 窗口 应 用 程序 可 以 拥有 羔 单 和 对 话 框 (这 对 一 个 程序 片 来 
说 是 不 可 能 的 和 很 困难 的 ) ， 可 是 如 果 我 们 使 用 一 个 老 版 本 的 Java， 我 
们 将 会 牺牲 本 地 操作 系统 环境 的 外 观 和 感受 。 正 C/Swing 库 允许 我 们 制 
造 一 个 保持 原来 操作 系统 环境 的 外 观 和 感受 的 应 用 程序 。 如 果 我 们 想 建 
立 一 个 设置 窗口 应 用 程序 ， 它 会 合理 地 运作 ， 同 样 ， 如 果 我 们 可 以 使 用 
最 新 版 本 的 Java 并 且 集 合 所 有 的 工具 ， 我 们 就 可 以 发 布 不 会 使 用 户 困 惑 
的 应 用 程序 。 如 果 因 为 一 些 原因 ， 我 们 被 迫使 用 老 版 本 的 Java， 请 在 毁 
坏 以 建立 重要 的 设置 窗口 的 应 用 程序 前 仔细 地 考虑 。 


13.15.1 菜单 


直接 在 程序 片 中 安放 一 个 某 单 是 不 可 能 的 〈Java 1.0,Javal.1 和 Swing 库 不 
人 允许) ， 因 为 它们 是 针对 应 用 程序 的 。 继 续 ， 如 果 您 不 相信 我 并 且 确 定 
在 程序 请 中 可 以 合理 地 拥有 沈 单 ， 那 么 您 可 以 去 试验 一 下 。 程 序 片 中 没 
有 setMenuBar() 方 法 ， 而 这 种 方法 是 附 在 菜单 中 的 (我 们 会 看 到 它 可 以 
合理 地 在 程序 片 产 生 一 个 帧 ， 并 且 帧 包含 菜单 )。 


有 四 种 不 同类 型 的 MenuComponent (菜单 组 件 ) ， 所 有 的 菜单 组 件 起 源 
于 抽象 类 : 沈 单 条 《我 们 可 以 在 一 个 事件 帧 里 拥有 一 个 菜单 条 ) ， 采 单 
去 文 配 一 个 单独 的 下 拉 菜 单 或 者 子 菜 单 、 菜 单项 来 说 明 菜 单 里 一 个 单个 
的 元 素 ， 以 及 起 源 于 Menultem, 产 生 检 查 标志 (checkmark)〉 去 显示 菜单 
项 是 否 被 选择 的 CheckBoxMenultem。 


不 同 的 系统 使 用 不 同 的 资源 ， 对 Java 和 AWT 而 言 ， 我 们 必须 在 源 代码 中 






































手工 汇编 所 有 的 沫 单 。 


//: Menu1. java 


// Menus work only with Frames. 
// Shows submenus, checkbox menu items 
// and swapping menus. 
import java.awt.*; 
public class Menu1 extends Frame { 
String[] flavors = { "Chocolate", "Strawberry", 
"Vanilla Fudge Swirl", "Mint Chip", 
"Mocha Almond Fudge", "Rum Raisin", 
"Praline Cream", "Mud Pie" }; 
TextField t = new TextField("No flavor", 30); 
MenuBar mb1 = new MenuBar(); 
Menu f = new Menu("File"); 
Menu m = new Menu("Flavors"); 
Menu s = new Menu("Safety"); 
// Alternative approach: 
CheckboxMenuItem[] safety = { 
new CheckboxMenuItem("Guard"), 
new CheckboxMenuItem("Hide" ) 
}; 


MenuItem[] file = { 


new MenulItem("Open"), 
new Menultem("Exit") 
}; 
// A second menu bar to swap to: 
MenuBar mb2 = new MenuBar(); 
Menu fooBar = new Menu("fooBar"); 
Menultem[] other = { 
new MenulItem("Foo"), 
new Menultem("Bar"), 
new MenulItem("Baz"), 
}; 
Button b = new Button("Swap Menus"); 
public Menui() { 
for(int i = 0; i < flavors.length; i++) { 
m.add(new MenuItem(flavors[i])); 
// Add separators at intervals: 
if((it+t1) % 3 == 0) 
m.addSeparator(); 
} 
for(int i = 0; i < safety.length; i++) 
s.add(safety[i]); 
f.add(s); 


for(int i = 0; i < file.length; i++) 


f.add(file[i]); 
mb1.add(f); 
mb1.add(m); 
setMenuBar (mb1) ; 
t.setEditable(false); 
add("Center", t); 
// Set up the system for swapping menus: 
add("North", b); 
for(int i = 0; i < other.length; i++) 
fooBar.add(other[i]); 
mb2.add(fooBar ) ; 
} 
public boolean handleEvent(Event evt) { 
if(evt.id == Event ,WINDOW_DESTROY ) 
System.exit(0); 
else 
return super.handleEvent(evt); 
return true; 
} 
public boolean action(Event evt, Object arg) 
if(evt.target.equals(b)) { 
MenuBar m = getMenuBar(); 


if(m == mb1) setMenuBar(mb2); 


else if (m == mb2) setMenuBar(mb1) ; 
} 
else if(evt.target instanceof MenuItem) { 
if(arg.equals("Open")) { 
String s = t.getText(); 
boolean chosen = false; 
for(int 1 = 0; i < flavors.length; i++) 
if(s.equals(flavors[i])) chosen = true; 
if(!chosen) 
t.setText("Choose a flavor first!"); 
else 
t.setText("Opening "+ s +". Mmm, mm!"); 
} 
else if(evt.target.equals(file[1])) 
System.exit(0); 
// CheckboxMenuItems cannot use String 
// matching; you must match the target: 
else if(evt.target.equals(safety[0])) 
t.setText("Guard the Ice Cream! " + 
"Guarding is " + safety[0].getState()); 
else if(evt.target.equals(safety[1])) 
t.setText("Hide the Ice Cream! " + 


"Is it cold? " + safety[1].getState()); 


else 
t.setText(arg.toString()); 
} 
else 
return super.action(evt, arg); 

return true; 

小 

public static void main(String[] args) { 
Menut f = new Menu1(); 
f.resize(300, 200); 
f.show(); 

} 


} ///:~ 





在 这 个 程序 中 ， 我 避免 了 为 每 个 亲 单 编写 典型 的 见长 的 add0 列 表 调 用 ， 

因为 那 看 起 来 像 许 多 的 无 用 的 标志 。 取 而 代 之 的 是 ， 我 安放 菜单 项 到 数 
组 中 ， 然 后 在 一 个 for 的 循环 中 通过 每 个 数组 调用 add0 简 单 地 跳 过 。 这 

样 的 话 ， 增 加 和 减少 亲 单 项 变 得 没 那 么 讨 大 了。 


作为 一 个 可 选择 的 方法 (我 发 现 这 很 难 令 我 满意 ， 因 为 它 需 要 更 多 的 分 
配 ) CheckboxMenuItems 在 数组 的 句柄 中 被 创建 是 被 称 为 安全 创建 ;这 
对 数组 文件 和 其 它 的 文件 而 言 是 真正 的 安全 。 


程序 中 创建 了 不 是 一 个 而 是 二 个 的 染 单 条 来 证 明 染 单条 在 程序 运行 时 能 
被 交换 激活 。 我 们 可 以 看 到 沫 单条 怎样 组 成 染 单 ， 每 个 菜单 怎样 组 成 菜 
单项 (Menultems) ，chenkboxMenuItems 或 者 其 它 的 菜单 〈 产 生子 菜 
单 ) 。 当 且 单 组 合 后 ， 可 以 用 setMenuBar0 方 法 安装 到 现在 的 程序 中 。 
值得 注意 的 是 当 按 钮 被 压 下 时 ， 它 将 检查 当前 的 染 单 安装 使 用 
getMenuBar0， 然 后 安放 其 它 的 逐 单条 在 它 的 位 置 上 。 




















当 测 试 是 “open”( 即 开始 ) 时 ， 注 意 拼 写 和 大 写 ， 如 果 开 始 时 没有 对 
象 ，Java 发 出 no error (没有 错误 ) 的 信号 。 这 种 字符 串 比 较 是 一 个 明显 
的 程序 设计 错误 源 。 


校 验 和 非 校 验 的 沫 单项 自动 地 运行 ， 与 之 相关 的 CheckBoxMenuItems 着 
实 令 人 吃惊 ， 这 是 因为 一 些 原 因 它 们 不 允许 字符 串 匹 配 。 (这 似乎 是 自 
相 矛 盾 的 ， 尺 管 字符 串 匹 配 并 不 是 一 种 很 好 的 办 法 。〉 因此， 我 们 可 以 
匹配 一 个 目标 对 象 而 不 是 它们 的 标签 。 当 演示 时 ，getState() 方 法 用 来 显 
示 状 态 。 我 们 同样 可 以 用 setState0 改 变 CheckboxMenuItem 的 状态 。 


我 们 可 能 会 认为 一 个 菜单 可 以 合理 地 置 入 超过 一 个 的 菜单 条 中 。 这 看 似 
合理 ， 因 为 所 有 我 们 忽略 的 沫 单条 的 add0) 方 法 都 是 一 个 句柄 。 然 而 ， 如 
果 我 们 试图 这 样 做 ， 这 个 结果 将 会 变 得 非常 的 别 捏 ， 而 远 非 我 们 所 希望 
得 到 的 结果 。 很 难 知道 这 是 一 个 编程 中 的 错误 或 者 说 是 他 们 试图 使 它 
以 这 种 方法 去 运行 所 产生 的 。〉 这 个 例子 同样 回 我 们 展示 了 为 什么 我 们 
需要 建立 一 个 应 用 程序 以 瞧 代 程序 请 。 (这 是 因为 应 用 程序 能 支持 羔 

单 ， 而 程序 片 是 不 能 直接 使 用 集 单 的 。) 我 们 从 帧 处 继承 代 蔡 从 程序 片 
处 继承 。 男 外 ， 我 们 为 类 建 一 个 构建 右 以 取代 init(0) 安 装 事 件 。 最 后 ， 我 
们 创建 一 个 main() 方 法 并 且 在 我 们 建 的 新 型 对 象 里 ， 调 整 它 的 大 小 ， 然 
后 调用 show()。 它 与 程序 片 只 在 很 小 的 地 方 有 不 同 之 处 ， 然 而 这 时 它 已 
经 是 一 个 独立 的 设置 窗口 应 用 程序 并 且 我 们 可 以 使 用 采 单 。 


13.15.2 对 话 框 


对 话 框 是 一 个 从 其 它 窗口 弹出 的 窗口 。 它 的 目的 是 处 理 一 些 特殊 的 争议 
和 它们 的 细节 而 不 使 原来 的 窗口 陷入 混乱 之 中 。 对 话 框 大 量 在 设置 窗口 
的 编程 环境 中 使 用 ， 但 惑 像 前 面 提 到 的 一 样 ， 鲜 于 在 程序 片 中 使 用 。 


我 们 需要 从 对 话 类 处 继承 以 创建 其 它 类 型 的 窗口 、 像 帧 一 样 的 对 话 框 。 

和 窗 杠 不同， 对 话 框 不 能 拥有 菜单 条 也 不 能 改变 光标 ， 但 除 此 之 外 它们 
十 分 的 相似 。 一 个 对 话 框 拥有 布局 管理 器 (默认 的 是 BorderLayout 布 局 
管理 器 ) 和 过 载 action0 等 等 ， 或 用 handleEventO 去 处 理事 件 。 我 们 会 注 
意 到 handleEventO 的 一 个 重要 差异 : 当 WINDOW_DESTORY 事 件 发 生 

时 ， 我 们 并 不 希望 关闭 正在 运行 的 应 用 程序 ! 


相反 ， 我 们 可 以 使 用 对 话 窗 口 通过 调用 dispace() 释 放 资 源 。 在 下 面 的 例 
子 中 ， 对 话 框 是 由 定义 在 那儿 作为 类 的 ToeButton 的 特殊 按钮 组 成 的 网 
格 构成 的 〈 利 用 GridLayonut 布 局 管理 器 ) 。ToeBnutton 按 钮 围绕 它 自 己 男 






































了 一 个 帧 ， 并 且 依 赖 它 的 状态 : 在 空 的 中 的 “X? 或 者 "0”。 它 从 空白 开 
台 ， 然 后 依靠 使 用 者 的 选择 ， 转 换 成 <X? 或 <0”。 但 是 ， 当 我 们 单 击 在 
按钮 上 时 ， 它 会 在 “X” 和 “0O” 之 间 来 回 交 换 。 这 产生 了 一 种 类 似 填 字 
游戏 的 感觉 ， 当 然 比 它 更 令 人 讨厌 。) 另外 ， 这 个 对 话 框 可 以 被 设置 为 
在 主 应 用 程序 窗口 中 为 很 多 的 行 和 列 变 更 号 码 。 


//: ToeTest.java 





// Demonstration of dialog boxes 

// and creating your own components 

import java.awt.*; 

class ToeButton extends Canvas { 
int state = ToeDialog.BLANK; 
ToeDialog parent; 
ToeButton(ToeDialog parent) { 

this.parent = parent; 

} 
public void paint(Graphics g) { 


int x1 = 0; 


int y1 = 0; 

int x2 = size().width - 1; 

int y2 = size().height - 1; 
g.drawRect(x1, yi, x2, y2); 
x1 = x2/4; 

yi = y2/4; 


int wide = x2/2; 


int high = y2/2; 
if(state == ToeDialog.Xx) { 
g.drawLine(x1, y1, x1 + wide, y1 + high); 
g.drawLine(x1, yi + high, x1 + wide, y1); 
} 
if(state == ToeDialog.00) { 


g.drawOval(xi, y1, xit+twide/2, yithigh/2); 


} 


public boolean 
mouseDown(Event evt, int x, int y) { 
if(state == ToeDialog.BLANK) { 
state = parent.turn; 
parent.turn= (parent.turn == ToeDialog.XX ? 
ToeDialog.00 : ToeDialog.XxX); 
} 
else 
state = (state == ToeDialog.XX ? 
ToeDialog.00 : ToeDialog. XxX); 
repaint(); 


return true; 


class ToeDialog extends Dialog { 
// w = number of cells wide 
// h = number of cells high 


static final int BLANK = 0; 


static final int XX 1; 


static final int 00 23 
int turn = XX; // Start with x's turn 
public ToeDialog(Frame parent, int w, int h) { 
super (parent, "The game itself", false); 
setLayout(new GridLayout(w, h)); 
for(int i = 0; i <w * h; i++) 
add(new ToeButton(this)); 
resize(w * 50, h * 50); 
} 
public boolean handleEvent(Event evt) { 
if(evt.id == Event ,WINDOW_DESTROY ) 
dispose(); 
else 


return super.handleEvent(evt); 


return true; 


} 


public class ToeTest extends Frame { 


TextField rows = new TextField("3"); 
TextField cols = new TextField("3"); 
public ToeTest() { 
setTitle("Toe Test"); 
Panel p = new Panel(); 
p.setLayout(new GridLayout(2,2)); 
p.add(new Label("Rows", Label.CENTER) ); 
p.add(rows); 
p.add(new Label("Columns", Label.CENTER) ); 
p.add(cols); 
add("North", p); 
add( "South", new Button("go")); 
} 
public boolean handleEvent(Event evt) { 
if(evt.id == Event ,WINDOW_DESTROY ) 
System.exit(0); 
else 
return super.handleEvent(evt); 
return true; 
} 
public boolean action(Event evt, Object arg) 
if(arg.equals("go")) { 


Dialog d = new ToeDialog( 


this, 
Integer.parseInt(rows.getText()), 
Integer.parseInt(cols.getText())); 
d.show(); 
} 
else 
return super.action(evt, arg); 
return true; 
} 
public static void main(String[] args) { 
Frame f = new ToeTest(); 
f.resize(200,100); 
f.show(); 
} 
} ///:~ 


ToeButton 类 保留 了 一 个 句柄 到 它 ToeDialog 型 的 父 类 中 。 正 如 前 面 所 
述 ，ToeButton 和 ToeDialog 高 度 的 结合 因为 一 个 ToeButton 只 能 被 一 个 
ToeDialog 所 使 用 ， 但 它 却 解决 了 一 系列 的 问题 ， 事 实 上 这 实在 不 是 一 
个 粳 糕 的 解雇 方案 因为 没有 另外 的 可 以 记录 用 户 选 择 的 对 话 类 。 当 然 我 
们 可 以 使 用 其 它 的 制造 ToeDialog.turn (ToeButton 的 静态 的 一 部 分 ) 方 
法 。 这 种 方法 消除 了 它们 的 紧密 联系 ， 但 却 阻止 了 我 们 一 次 拥有 多 个 
ToeDialog (无 论 如 何 ， 至 少 有 一 个 正常 地 运行 )。 


paintO 是 一 种 与 图 形 有 关 的 方法 : 它 围 绕 按钮 男 出 定形 并 画 
出 “X” 或 “<O”。 这 完全 是 见长 的 计算 ， 但 却 十 分 的 直观 。 


一 个 鼠标 单 击 被 过 载 的 mouseDown() 方 法 所 俘获 ， 最 要 紧 的 是 检查 是 否 





有 事件 写 在 按钮 上 。 如 果 没 有 ， 父 窗口 会 被 询问 以 找 出 谁 选择 了 它 并 用 
来 确定 按钮 的 状态 。 值 得 注意 的 是 按钮 随后 交 回 到 父 类 中 并 且 改 变 它 的 
选择 。 如 果 按 钮 已 经 显示 这 为 “<X” 和 “OO” 那么 它们 会 被 改变 状态 。 我 
们 能 注意 到 本 书 第 三 章 中 描述 的 在 这 些 计算 中 方便 的 使 用 的 三 个 一 组 的 
If-else。 当 一 个 按钮 的 状态 改变 后 ， 按 钮 会 被 重 画 。 


ToeDialog 的 构建 器 十 分 的 简单 : 它 像 我 们 所 需要 的 一 样 增加 一 些 按钮 
到 GridLayout 布 局 管理 器 中 ， 然 后 调整 每 个 按钮 每 边 大 小 为 50 个 像素 
《如 果 我 们 不 调整 窗口 ， 那 么 它 束 不 会 显示 出 来 ) 。 注 意 handleEvent() 
正好 为 WINDOW_DESTROY 调 用 dispose()， 因 此 整个 应 用 程序 不 会 被 关 
闭 。 


ToeTest 设 置 整个 应 用 程序 以 创建 TextField (为 输入 按钮 网 格 的 行 和 列 》 
和 “go” 按 钮 。 我 们 会 领会 action0 在 这 个 程序 中 使 用 不 太 令 人 满意 的 “ 字 
符 串 匹配 ”技术 来 测试 按钮 的 按 下 (请 确定 我 们 拼写 和 大 写 都 是 正确 
的 ! ) 。 当 按钮 按 下 时 ，TextField 中 的 数据 将 被 取出 ， 并 且 ， 因 为 它们 
在 字符 串 结 构 中 ， 所 以 需要 利用 静态 的 Integer.paresInt0) 方 法 来 转变 成 中 
断 。 一 旦 对 话 类 被 建立 ， 我 们 就 必须 调用 show0) 方 法 来 显示 和 激活 它 。 


我 们 会 注意 到 ToeDialog 对 象 赋 值 给 一 个 对 话 句 柄 d。 这 是 一 个 上 漳 造 型 
的 例子 ， 尽 管 它 没有 真正 地 产生 重要 的 差异 ， 因 为 所 有 的 事件 都 是 
show() 调 用 的 。 但 是 ， 如 果 我 们 想 调用 ToeDialog 中 已 经 存在 的 一 些 方 
法 ， 我 们 需要 对 ToeDialog 句 柄 赋值 ， 就 不 会 在 一 个 上 渊 中 丢失 信息 。 


1. 文件 对 话 类 


在 一 些 操作 系统 中 拥有 许多 的 特殊 内 建 对 话 框 去 处 理 选择 的 事件 ， 例 
如 : 人 字库， 颜色 ， 打 印 机 以 及 类 似 的 事件 。 几 乎 所 有 的 操作 系统 都 支持 
打开 和 保存 文件 ， 但 是 ，Java 的 FileDialog 包 更 容易 使 用 。 当 然 这 会 不 再 
检测 所 有 使 用 的 程序 片 ， 因 为 程序 片 在 本 地 磁盘 上 既 不 能 读 也 不 能 写 文 
件 。《〈 这 会 在 新 的 浏览 器 中 交换 程序 片 的 信任 关系 。) 

下 面 的 应 用 程序 运用 了 两 个 文件 对 话 类 的 窗 体 ， 一 个 是 打开 ， 一 个 是 保 
存 。 大 多 数 的 代码 到 如 今 已 为 我 们 所 熟悉 ， 而 所 有 这 些 有 趣 的 活动 及 生 
在 两 个 不 同 按钮 单 击 事 件 的 action0 方 法 中 。 


//: FileDialogTest.java 























// Demonstration of File dialog boxes 
import java.awt.*; 
public class FileDialogTest extends Frame { 
TextField filename = new TextField(); 
TextField directory = new TextField(); 
Button open = new Button("Open"); 
Button save = new Button("Save"); 
public FileDialogTest() { 
setTitle("File Dialog Test"); 
Panel p = new Panel(); 
p.setLayout(new FlowLayout()); 
p.add(open); 
p.add(save); 
add("South", p); 
directory.setEditable(false); 
filename.setEditable(false); 
p = new Panel(); 
p.setLayout(new GridLayout(2,1)); 
p.add( filename) ; 
p.add(directory); 
add("North", p); 


} 


public boolean handleEvent(Event evt) { 


if(evt.id == EVent ,WINDOW_DESTROY ) 
System.exit(0); 
else 
return super.handleEvent(evt); 
return true; 
} 
public boolean action(Event evt, Object arg) { 
if(evt.target.equals(open)) { 
// Two arguments, defaults to open file: 
FileDialog d = new FileDialog(this, 

"What file do you want to open?"); 
d.setFile("*.java"); // Filename filter 
d.setDirectory("."); // Current directory 
d.show(); 

String openFile; 

if((openFile = d.getFile()) != null) { 
filename.setText(openFile) ; 
directory.setText(d.getDirectory()); 

} else { 
filename.setText("You pressed cancel"); 


directory.setText(""); 


else if(evt.target.equals(save)) { 

FileDialog d = new FileDialog(this, 
"What file do you want to save?", 
FileDialog.SAVE); 

d.setFile("*.java"); 

d.setDirectory("."); 

d.show(); 

String saveFile; 

if((saveFile = d.getFile()) != null) { 
filename.setText(saveFile) ; 
directory.setText(d.getDirectory()); 

} else { 
filename.setText("You pressed cancel"); 


directory.setText(""); 


} 


else 
return super.action(evt, arg); 
return true; 
} 
public static void main(String[] args) { 
Frame f = new FileDialogTest(); 


f.resize(250,110); 


f.show(); 


} 
FA ss 








对 一 个 “打开 文件 ”对 话 框 ， 我 们 使 用 构建 器 设置 两 个 自 变 量 ;， 首先 是 父 
窗口 句柄 ， 其 次 是 FileDialog 标 题 条 的 标题 。setFile() 方 法 提供 一 个 初始 
文件 名 一 一 也 许 本 地 操作 系统 文 持 通配符 ， 因 此 在 这 个 例子 中 所 有 

的 .java 文 件 最 开头 会 被 显示 出 来 。setDirectory() 方 法 选择 文件 决定 开始 
的 目录 (一 般 而 言 ， 操 作 系 统 允 许 用 户 改 变 目 录 ) o 


show() 命 令 直到 对 话 类 关闭 才 返 回 。FileDialog 对 象 一 直 存 在 ， 因 此 我 们 
可 以 从 它 那 里 读 取 数据 。 如 果 我 们 调用 getFile0 并 且 它 返回 空 ， 这 意味 
着 用 户 退 出 了 对 话 类 。 文 件 名 和 调用 getDirectory0 方 法 的 结果 都 显示 在 
TextFields 里 。 


按钮 的 保存 工作 使 用 同样 的 方法 ， 除 了 因为 FileDialog 而 使 用 不 同 的 构 
建 器 。 这 个 构建 器 设置 了 三 个 自 变 量 并 且 第 三 的 一 个 自 变 量 必须 为 
FileDialog.SAVE 或 FileDialog.OPEN。 














13.16 新 型 AWT 


在 Java 1.1 中 一 个 显著 的 改变 束 是 完善 了 新 AWT 的 创新 。 大 多 数 的 改变 
围绕 在 Java 1.1 中 使 用 的 新 事件 模型 : 老 的 事件 模型 是 糟糕 的 、 笨 拙 
的 、 非 面向 对 象 的 ， 而 新 的 事件 模型 可 能 是 我 所 见 过 的 最 优秀 的 。 难 以 
理解 一 个 如 此 糟糕 的 〈 老 的 AWT) 和 一 个 如 此 优秀 的 〈 新 的 事件 模 
型 ) 程序 语言 居然 出 自 同一 个 集团 之 手 。 新 的 考虑 事件 的 方法 看 来 中 止 
了 ， 因 此 争议 不 再 变 成 障碍 ， 从 而 轻易 进入 我 们 的 意识 里 ， 相 反 ， 它 是 
一 个 帮助 我 们 设计 系统 的 工具 。 它 同样 是 Java Beans 的 精华 ， 我 们 会 在 
本 章 后 面部 分 进入 讲述 。 


新 的 方法 设计 对 象 做 为 “事件 源 ” 和 "事件 接收 器 ”以 代 蔡 老 AWT 的 非 面 向 
对 象 串 联 的 条 件 语句 。 正 象 我 们 将 看 到 的 内 部 类 的 用 途 古 集 成 面 同 对 象 
的 原始 状态 的 新 事件 。 另 外 ， 事 件 现在 被 描绘 为 在 一 个 类 体系 以 取代 单 
一 的 类 并 且 我 们 可 以 创建 自己 的 事件 类 型 。 


我 们 同样 会 发 现 ， 如 果 我 们 采用 老 的 AWT 编 程 ，Java 1.1 版 会 产生 一 些 
看 起 来 不 合理 的 名 字 转 换 。 例 如 ，setsize() 改 成 resize()。 当 我 们 学 习 Java 
Beans 时 这 会 变 得 更 加 的 合理 ， 因 为 Beans 使 用 一 个 独特 的 命名 协议 。 名 
字 必 须 被 修改 以 在 Beans 中 产生 新 的 标准 AWT 组 件 。 


雯 贴 板 操 作 在 Java 1.1 厂 中 也 得 到 文 持 ， 尽 管 拖 放 操 作 “ 将 在 新 版 本 中 被 
支持 ”。 我 们 可 能 访问 桌面 色彩 组 织 ， 所 以 我 们 的 Java 可 以 同 其 余 桌 面 
保持 一 致 。 可 以 利用 弹出 式 染 单 ， 并 且 为 图 像 和 图 形 作 了 改进 。 也 同样 
支持 鼠标 操作 。 还 有 简单 的 为 打印 的 API 以 及 简单 地 文 持 深 动 。 


13.16.1 新 的 事件 模型 


在 新 的 事件 模型 的 组 件 可 以 开始 一 个 事件 。 每 种 类 型 的 事件 被 一 个 个 别 
的 类 所 描绘 。 当 事件 开始 后 ， 它 受理 一 个 或 更 多 事件 指明 "接收 器 ”。 因 
此 ， 事 件 源 和 处 理事 件 的 地 址 可 以 被 分 离 。 


每 个 事件 接收 器 都 是 执行 特定 的 接收 器 类 型 接口 的 类 对 象 。 因 此 作为 一 
个 程序 开发 者 ， 我 们 所 要 做 的 是 创建 接收 器 对 象 并 且 在 被 激活 事件 的 组 
件 中 进行 注册 。event-firing 组 件 调 用 一 个 addXXXListener() 方 法 来 完成 

注册 ， 以 描述 XXX 事件 类 型 接受 。 我 们 可 以 容易 地 了 解 到 以 addListened 














名 的 方法 通知 我 们 任何 的 事件 类 型 都 可 以 被 处 理 ， 如 果 我 们 试图 接收 事 
件 我 们 会 发 现 编 译 时 我 们 的 错误 。Java Beans 同 样 使 用 这 种 addListener 名 
的 方法 去 判断 那 一 个 程序 可 以 运行 


我 们 所 有 的 事件 逻辑 将 装 入 到 一 个 接收 右 类 中 。 当 我 们 创建 一 个 接收 费 
类 时 唯一 的 一 点 限制 是 必须 执行 专用 的 接口 。 我 们 可 以 创建 一 个 全 局 接 
收 右 类 ， 这 种 情况 在 内 部 类 中 有 助 于 被 很 好 地 使 用 ， 不 仅仅 是 因为 它们 
提供 了 一 个 理论 上 的 接收 絮 类 组 到 它们 服务 的 UI 或 业务 逻辑 类 中 ， 但 因 
为 〈《 正 像 我们 将 会 在 本 章 后 面 看 到 的 ) 事实 是 一 个 内 部 类 维持 一 个 句柄 
到 它 的 父 对 象 ， 提 供 了 一 个 很 好 的 通过 类 和 子 系 统 边界 的 调用 方法 。 


一 个 简单 的 例子 将 使 这 一 切 变 得 清晰 明确 。 同 时 思考 本 章 前 部 
Button2.java 例 子 与 这 个 例子 的 差异 。 


//: Button2New. java 








// Capturing button presses 
import java.awt.*; 
import java.awt.event.*; // Must add this 
import java.applet.*; 
public class Button2New extends Applet { 
Button 
b1 = new Button("Button 1"), 
b2 = new Button("Button 2"); 
public void init() { 
b1.addActionListener(new B1()); 
b2.addActionListener(new B2()); 
add(b1); 


add(b2); 


} 


class B1 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


getAppletContext().showStatus("Button 1"); 


} 


class B2 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


getAppletContext().showStatus("Button 2"); 


} 


/* The old way: 
public boolean action(Event evt, Object arg) { 
if(evt.target.equals(b1) ) 
getAppletContext().showStatus("Button 1"); 
else if(evt.target.equals(b2) ) 
getAppletContext().showStatus("Button 2"); 
// Let the base class handle it: 
else 
return super.action(evt, arg); 


return true; // We've handled it here 


a 


} ///:~ 





我 们 可 比较 两 种 方法 ， 老 的 代码 在 左面 作为 注解 。 在 init() 方 法 里 ， 只 有 
一 个 改变 就 是 增加 了 下 面 的 两 行 : 


b1.addActionListener(new B1()); 





b2.addActionListener(new B2()); 


按钮 按 下 时 ，addActionListener() 通 知 按钮 对 象 被 激活 。B1l 和 B2 类 都 是 
执行 接口 ActionListener 的 内 部 类 。 这 个 接口 包括 一 个 单一 的 方法 
actionPerformed()〈( 这 意味 着 当 事 件 激活 时 ， 这 个 动作 将 被 执行 )。 注 
意 actionPreformed() 方 法 不 是 一 个 普通 事件 ， 说 得 更 恰当 些 是 一 个 特殊 
类 型 的 事件 ，ActionEvent。 如 果 我 们 想 提 取 特 殊 ActionEvent 的 信息 ， 
此 我 们 不 需要 故意 去 测试 和 下 浏 造型 自 变 量 。 


对 编程 者 来 说 一 个 最 好 的 事 便 是 actionPerformed0 十 分 的 简单 易 用 。 它 
是 一 个 可 以 调用 的 方法 。 同 老 的 action(0) 方 法 比较 ， 老 的 方法 我 们 必须 指 
出 发 生 了 什么 和 适当 的 动作 ， 同 样 ， 我 们 会 担心 调用 基础 类 action() 的 版 
本 并 且 返 回 一 个 值 去 指明 是 否 被 处 理 。 在 新 的 事件 模型 中 ， 我 们 知道 所 
有 事件 测试 推理 自动 进行 ， 因 此 我 们 不 必 指 出 发 生 了 什么 ; 我 们 刚刚 表 
示 发 生 了 什么 ， 它 就 自动 地 完成 了 。 如 果 我 们 还 没有 提出 用 新 的 方法 履 
盖 老 的 方法 ， 我 们 会 很 快 提出 。 

13.16.2 事件 和 接收 者 类 型 

所 有 AWT 组 件 都 被 改变 成 包 仿 addXXXListener() 和 removeXXXListener() 
方法 ， 因 此 特定 的 接收 器 类 型 可 从 每 个 组 件 中 增加 和 删除 。 我 们 会 注意 
到 “XXX” 在 每 个 场合 中 同样 表示 自 变 量 的 方法 ， 例 如 ， 
addFooListener(FooListener fl))。 下 面 这 张 表格 总 结 了 通过 提供 


addXXXListener() 和 removeXXXListener() 方 法 ， 从 而 支持 那些 特定 事件 
的 相关 事件 、 接 收 器 、 方 法 以 及 组 件 。 


事件 ， 接 收 器 接口 及 添加 和 删除 方法 支持 这 个 事件 的 组 件 


ven listener interface andl | 






































ik and remove-methods ey supporting this event | 


‘ActionEvent 
ActionListener 
addActionListener( ) 


removeActionListener( ) 


AdjustmentEvent 
AdjustmentListener 
addAdjustmentListener( ) 


removeAdjustmentListener( ) 


ComponentEvent 
ComponentListener 
addComponentListener( ) 


removeComponentListener( ) 


ContainerEvent 
ContainerListener 
addContainerListener( ) 


removeContainerListener( ) 


FocusEvent 


FocusListener 


Button, List, TextField, Menultem, and 
its derivatives i i 
CheckboxMenultem, 
PopupMenu 


Menu, 


Scrollbar 


Anything you create that implements the 
Adjustable interface 


Component and its derivatives, including 


Button, Canvas, Checkbox, Choice, 
Container, Panel, Applet, ScrollPane, 
Window, Dialog, FileDialog, Frame, 
Label, List, Scrollbar, TextArea, and 
TextField 


Container and its derivatives, including 
Panel, Applet, ScrollPane, Window, 
Dialog, FileDialog, and Frame 





Component and its derivatives, including 
Button, Canvas, Checkbox, Choice, 
Container, Panel, Applet, ScrollPane, 


addFocusListener( ) Window, Dialog, FileDialog, Frame 


Label, List, Scrollbar, TextArea, and 
removeFocusListener( ) TextField 


KeyEvent Component and its derivatives, including 
Button, Canvas, Checkbox, Choice, 
KeyListener Container, Panel, Applet, ScrollPane, 
Window, Dialog, FileDialog, Frame, 
addKeyL istener‘( ) Label, List, Scrollbar, TextArea, and 


À TextField 
removeKeyListener( ) 


MouseEvent (for both clicks andlComponent and its derivatives, including 
motion) Button, Canvas, Checkbox, Choice, 

Container, Panel, Applet, ScrollPane, 
MouseListener Window, Dialog, FileDialog, Frame, 


Label, List, Scrollbar, TextArea, and 
addMouseListener( ) TextField 


removeMouseListener( ) 


MouseEvent[55] (for both clicks||Component and its derivatives, including 
and motion) Button, Canvas, Checkbox, Choice, 

Container, Panel, Applet, ScrollPane, 
MouseMotionListener Window, Dialog, FileDialog, Frame, 


Label, List, Scrollbar, TextArea, and 
addMouseMotionListener()  ||TextField 


removeMouseMotionListener( ) 


WindowEvent Window and its derivatives, including 
Dialog, FileDialog, and Frame 

WindowListener 

addWindowListener( ) 


removeWindowListener( ) 





ItemEvent Checkbox, CheckboxMenultem, 
Choice, List, and anything that 
ItemListener implements the ItemSelectable interface 


addItemListener( ) 


removeltemListener( ) 


TextEvent Anything derived from TextComponent, 
including TextArea and TextField 


TextListener 
addTextListener( ) 


removeTextListener( ) 





©): 即使 表面 上 如 此 ， 但 实际 上 并 没有 MouseMotiionEvent (鼠标 运动 
事件 ) 。 单 击 和 运动 都 合成 到 MouseEvent 里 ， 所 以 MouseEvent 在 表格 中 
的 这 种 另类 行为 并 非 一 个 错误 。 


可 以 看 到 ， 每 种 类 型 的 组 件 只 为 特定 类 型 的 事件 提供 了 文 持 。 这 有 助 于 
我 们 发 现 由 每 种 组 件 支持 的 事件 ， 如 下 表 所 示 : 


组 件 类 型 支持 的 事件 


Component type Events supported by this component 
Adjustable AdjustmentEvent 


Applet ContainerEvent， FocusEvent, KeyEvent, 
MouseEvent, ComponentEvent 


Button 








ActionEvent, FocusEvent, KeyEvent, 
MouseEvent, ComponentEvent 


| TT | 





Canvas FocusEvent, KeyEvent, MouseEvent, 
ComponentEvent 


Checkbox ItemEvent, FocusEvent, KeyEvent, MouseEvent, 


ComponentEvent 


ItemEvent, FocusEvent, KeyEvent, MouseEvent, 
ComponentEvent 
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Choice 


Component FocusEvent, KeyEvent, MouseEvent, 


ComponentEvent 


ContainerEvent, FocusE vent, KeyEvent, 
MouseEvent, ComponentEvent 

Dialog ContainerEvent, WindowEvent, FocusEvent, 
KeyEvent, MouseEvent, ComponentEvent 


ContainerEvent, WindowEvent, FocusEvent, 
KeyEvent, MouseEvent, ComponentEvent 

Frame ContainerEvent, WindowEvent,  FocusEvent, 
KeyEvent, MouseEvent, ComponentEvent 

Label FocusEvent, KeyEvent, MouseEvent, 
ComponentEvent 


List 


ActionEvent, FocusEvent, KeyEvent, 
MouseEvent, ItemEvent, ComponentEvent 


Menu ‘ActionEvent 
‘ActionEvent 
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[Panel | | 


ContainerEvent, FocusEvent, 
KeyEvent, MouseEvent, 
ComponentEvent 


Scrollbar AdjustmentEvent, FocusEvent, 
KeyEvent, MouseEvent, 
ComponentEvent 

ScrollPane ContainerEvent, FocusEvent, 
KeyEvent, MouseEvent, 
ComponentEvent 


'TextArea ‘TextEvent, FocusEvent, KeyEvent, 
MouseEvent, ComponentEvent 


'TextComponent TextEvent, FocusEvent, KeyEvent, 
MouseEvent, ComponentEvent 


ActionEvent, TextEvent, FocusEvent, 
KeyEvent, MouseE vent, 
ComponentEvent 


ContainerEvent, WindowEvent, 
FocusEvent, KeyEvent, MouseEvent, 
ComponentEvent 





一 旦 知道 了 一 个 特定 的 组 件 文 持 哪 些 事 件 ， 就 不 必 再 去 寻找 任何 东西 来 
啊 应 那个 事件 。 只 需 简单 地 : 


(1) 取得 事件 类 的 名 字 ， 并 删 掉 其 中 的 “Event" 字 样 。 在 剩 下 的 部 分 加 





入 “Listener” 字 样 。 这 束 是 在 我 们 的 内 部 类 里 需要 实现 的 接收 絮 接口 。 


(2) 实现 上 面 的 接口 ， 针 对 想 要 捕获 的 事件 编写 方法 代码 。 例 如 ， 假 设 
我 们 想 捕 获 女 标的 移动 ， 所 以 需要 为 MouseMotiionListener 接 口 的 
mouseMoved0) 方 法 编写 代 〈 当 然 还 必须 实现 其 他 一 些 方法 ， 但 这 里 有 捷 
径 可 循 ， 马 上 束 会 讲 到 这 个 问题 ) 。 


(3) 为 步骤 2 中 的 接收 器 类 创建 一 个 对 象 。 随 自己 的 组 件 和 方法 完成 对 它 
的 注册 ， 方 法 是 在 接收 缉 的 名 字 里 加 入 一 个 前 缀 “add"。 比 如 
addMouseMotionListener(). 

下 表 是 对 接收 器 接口 的 一 个 总 结 : 


PW Cae A 接口 中 的 方法 





Listener interface 


Methods in interface 





w/ adapter 


ActionListener actionPerformed(ActionEvent) 


AdjustmentListener |jadjustmentValueChanged( 
AdjustmentEvent) 

ComponentListener |icomponentHidden(ComponentEvent) 

ComponentAdapter |icomponentShown(ComponentEvent) 
componentMoved(ComponentEvent) 


componentResized(ComponentEvent) 


ContainerListener componentAdded(ContainerEvent) 


ContainerAdapter componentRemoved(ContainerEvent) 





FocusListener 
FocusAdapter 
KeyListener 


KeyAdapter 


MouseListener 


MouseAdapter 


MouseMotionListener 


MouseMotionAdapter 


WindowListener 


W indowAdapter 


focusGained(FocusEvent) 


focusLost(FocusEvent) 


keyPressed(KeyEvent) 
keyReleased(KeyEvent) 

key Typed(KeyEvent) 
mouseClicked(MouseEvent) 
mouseEntered(MouseEvent) 
mouseExited(MouseEvent) 
mousePressed(MouseEvent) 


mouseReleased(MouseEvent) 


mouseDragged(MouseEvent) 


mouseMoved(MouseEvent) 


windowOpened(WindowEvent) 


windowClosing(WindowEvent) 


windowClosed(WindowEvent) 
windowActivated(WindowEvent) 
windowDeactivated(WindowEvent) 
windowIconified(WindowEvent) 


windowDeiconified(WindowEvent) 





|ItemListener itemStateChanged(ItemE vent) | 


text ValueChanged(TextEvent) 


1. 用 接收 器 适 配 需 简化 操作 


在 上 面 的 表格 中 ， 我 们 可 以 注意 到 一 些 接收 器 接口 只 有 唯一 的 一 个 方 

法 。 它 们 的 执行 是 无 轻重 的 ， 因 为 我 们 仅 当 需要 书写 特殊 方法 时 才 会 执 
行 它们 。 然 而 ， 接 收 器 接口 拥有 多 个 方法 ， 使 用 起 来 却 不 太 友好 。 例 

如 ， 我 们 必须 一 直 运 行 某 些 事物 ， 当 我 们 创建 一 个 应 用 程序 时 对 帧 提供 
一 个 WindowListener， 以 便当 我 们 得 到 windowClosing0 事 件 时 可 以 调用 
System.exit(0) 以 退出 应 用 程序 。 但 因为 WindowListener 是 一 个 接口 ， 我 
们 必须 执行 其 它 所 有 的 方法 即使 它们 不 运行 任何 事件 。 这 真 令 人 讨厌 。 


为 了 解决 这 个 问题 ， 每 个 拥有 超过 一 个 方法 的 接收 器 接口 都 可 拥有 适 配 
上 器， 它们 的 名 我 们 可 以 在 上 面 的 表格 中 看 到 。 每 个 适配器 为 每 个 接口 方 
法 提供 默认 的 方法 。 (WindowAdapter 的 默认 方法 不 是 
windowClosing()， 而 是 System.exit(0) 方 法 。) 此 外 我 们 所 要 做 的 就 是 从 
适配器 处 继承 并 过 载 唯一 的 需要 变更 的 方法 。 例 如 ， 典 型 的 
WindowListener 我 们 会 像 下 面 这 样 的 使 用 。 


class MyWindowListener extends WindowAdapter { 











public void windowClosing(WindowEvent e) { 


System.exit(0); 
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但 所 谓 的 “适配器 ?也 有 一 个 缺点 ， 而 且 较 难 发 党 。 假 定 我 们 象 上 面 那样 
写 一 个 WindowAdapter: 


class MyWindowListener extends WindowAdapter { 


public void WindowClosing(WindowEvent e) { 


System.exit(0); 





表面 上 一 切 正常 ， 但 实际 没有 任何 效果 。 每 个 事件 的 编译 和 运行 都 很 正 
党 只 是 关闭 窗口 不 会 退出 程序 。 您 注意 到 问题 在 哪里 吗 ? 在 方法 的 
名 字 里 : 是 WindowClosing()， 而 不 是 windowClosing()。 大 小 写 的 一 个 
简单 失误 就 会 造成 一 个 轿 新 的 方法 。 但 是 ， 这 并 非 我 们 关闭 窗口 时 调用 
的 方法 ， 所 以 当然 没有 任何 效果 。 


13.16.3 用 Java 1.1 AWT 制 作 窗 口 和 程序 片 


我 们 经 常 都 需 i 要 创建 一 个 关 ne 个 窗口 调用 ， 亦 可 作为 一 
， 为 做 到 这 一 点 ， 内 需 为 程序 片 简单 地 加 入 一 个 mainO) 妈 
可 ， 令 其 在 一 个 Frame Cin) 里 构建 程序 片 的 一 上 实例。 作为 一 个 简单 
的 示例 |， 下 面 让 我 们 来 看 看 如 何 对 Button2New.java 作 一 番 修 改 ， 使 其 能 
同时 作为 应 用 程序 和 程序 片 使 用 : 


//: Button2NewB. java 

















// An application and an applet 

import java.awt.*; 

import java.awt.event.*; // Must add this 

import java.applet.*; 

public class Button2NewB extends Applet { 
Button 


b1 = new Button("Button 1"), 


b2 = new Button("Button 2"); 
TextField t = new TextField(20); 
public void init() { 

b1.addActionListener(new B1()); 

b2.addActionListener(new B2()); 
add(b1); 

add(b2); 

add(t); 

} 
class B1 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


t.setText("Button 1"); 


J 


class B2 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


t.setText("Button 2"); 


} 


// To close the application: 
static class WL extends WindowAdapter { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


} 


// A main() for the application: 

public static void main(String[] args) { 
Button2NewB applet = new Button2NewB(); 
Frame aFrame = new Frame("Button2NewB"); 
aFrame.addWindowListener(new WL()); 
aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true) ; 


} 
} ///:~ 





内 部 类 WL 和 main() 方 法 是 加 入 程序 片 的 唯一 两 个 元 素 ， 程 序 片 剩余 的 
部 分 则 原封 未 动 。 事 实 上 ， 我 们 通常 将 WL 类 和 main() 方 法 做 一 结 小 的 
改进 复制 和 粘贴 到 我 们 上 自己 的 程序 片 里 (请 记 住 创建 内 部 类 时 通常 需要 
个 外 部 类 来 处 理 它 ， 形 成 它 静 态 地 消除 这 个 需要 ) 。 我 们 可 以 看 到 在 
main( 方 法 里 ， 程 序 片 明确 地 初始 化 和 开始 ， 因 为 在 这 个 例子 里 浏览 器 
不 能 为 我 们 有 效 地 运行 它 。 当 然 ， 这 不 会 提供 全 部 的 浏览 器 调用 stop( 
和 destroy0 的 行为 ， 但 对 大 多 数 的 情况 而 言 它 都 是 可 接受 的 。 如 果 它 变 
成 一 个 拱 烦 ， 我 们 可 以 : 


(1) 使 程序 片 句柄 为 一 个 静态 类 《以 代 蔡 局 部 可 变 的 main0) ， 人 然后: 











(2) 在 我 们 调用 System.exit(0) 之 前 在 WindowAdapter.windowClosing() 中 调 
用 applet.stopO 和 applet.destroy()。 


注意 最 后 一 行 : 
aFrame.setVisible(true); 


这 是 Java 1.1 AWT 的 一 个 改变 。show() 方 法 不 再 被 支持 ， 而 
setVisible(true) 则 取代 了 show() 方 法 。 当 我 们 在 本 章 后 面部 分 学 习 Java 
Beans 时 ， 这 些 表面 上 易于 改变 的 方法 将 会 变 得 更 加 的 合理 。 


这 个 例子 同样 被 使 用 TextField 修 改 而 不 是 显示 到 控制 台 或 浏览 器 状态 行 
上 。 在 开发 程序 时 有 一 个 限制 条 件 就 是 程序 片 和 应 用 程序 我 们 都 必须 根 
据 它 们 的 运行 情况 选择 输入 和 输出 结构 。 

这 里 展示 了 Java 1.1 AWT 的 其 它 小 的 新 功能 。 我 们 不 再 需要 去 使 用 有 错 
误 倾 向 的 利用 字符 串 指定 BorderLayout 定 位 的 方法 。 当 我 们 增加 一 个 元 
素 到 Java 1.1 版 的 BorderLayout 中 时 ， 我 们 可 以 这 样 写 : 


aFrame.add(applet, BorderLayout. CENTER); 

我 们 对 位 置 规定 一 个 BorderLayout 的 常数 ， 以 使 它 能 在 编译 时 被 检验 
《而 不 是 对 老 的 结构 悄悄 地 做 不 合适 的 事 ) 。 这 是 一 个 显著 的 改善 ， 并 
且 将 在 这 本 书 的 余下 部 分 大 量 地 使 用 。 

2. 将 帘 口 接收 终 变 成 匿名 类 

任何 一 个 接收 器 类 都 可 作为 一 个 匿名 类 执行 ， 但 这 一 直 有 个 意外 ， 那 就 
是 我 们 可 能 需要 在 其 它 场合 使 用 它们 的 功能 。 但 是 ， 窗 口 接收 器 在 这 里 
仅 作 为 关闭 应 用 程序 窗口 来 使 用 ， 因 此 我 们 可 以 安全 地 制造 一 个 匿名 
类 。 然 后 ，main() 中 的 下 面 这 行 代码 : 

aFrame.addWindowListener(new WL()); 

会 变 成 : 


aFrame.addwindowListener( 


























new WindowAdapter() { 


public void windowClosing(WindowEvent e) { 


System.exit(0); 
} 
+); 





这 有 一 个 优点 就 是 它 不 需要 其 它 的 类 名 。 我 们 必须 对 目 己 判断 是 否 它 使 
代码 变 得 易于 理解 或 者 更 难 。 不 过 ， 对 本 书 余下 部 分 而 言 ， 匿 名 内 部 类 
将 通常 被 使 用 在 窗口 接收 需 中 。 


3. 将 程序 片 封装 到 JAR 文 件 里 


一 个 重要 的 JAR 应 用 残 是 完善 程序 片 的 装载 。 在 Java 1.0 版 中 ， 人 们 倾 回 
于 试 法 将 它们 的 代码 填 入 到 单个 的 程序 片 类 里 ， 因 此 客户 只 需要 单个 的 
服务 器 就 可 适合 下 载 程 序 片 代码 。 但 这 不 仅 使 结果 凑 乱 ， 难 以 阅读 《〈 当 
然 维护 也 然 ) 程序 ， 但 类 文件 一 直 不 能 压缩 ， 因 此 下 载 从 来 没有 快 过 。 


JAR 文 件 将 我 们 所 有 的 被 压缩 的 类 文件 打包 到 一 个 单个 儿 的 文件 中 ， 再 
被 浏览 器 下 载 。 现 在 我 们 不 需要 创建 一 个 糟糕 的 设计 以 最 小 化 我 们 创建 
的 类 ， 并 且 用 户 将 得 到 更 快 地 下 载 速度 。 


仔细 想 想 上 面 的 例子 ， 这 个 例子 看 起 来 像 Button2NewB， 是 一 个 单 类 ， 
但 事实 上 它 包 含 三 个 内 部 类 ， 因 此 共有 四 个 。 每 当 我 们 编译 程序 ， 我 会 
用 这 行 代码 打包 它 到 一 个 JAR 文 件 : 








jar cf Button2NewB .jar *.class 


这 是 假定 只 有 一 个 类 文件 在 当前 目录 中 ， 其 中 之 一 来 自 
Button2NewB.java〈 和 否则 我 们 会 得 到 特别 的 打包 ) 。 


N 以 创建 一 个 使 用 新 文件 标签 来 指定 JAR 文 件 的 HTML 页 ， 如 
下 所 示 : 


<head><title>Button2NewB Example Applet 














</title></head> 


<body> 


<applet code="Button2NewB.class" 
archive="Button2NewB. jar" 
width=200 height=150> 
</applet> 


</body> 


与 HTML 文 件 中 的 程序 片 标记 有 关 的 其 他 任何 内 容 都 保持 不 变 。 
13.16.4 再 研究 一 下 以 前 的 例子 


为 注意 到 一 些 利用 新 事件 模型 的 例子 和 为 学 习 程序 从 老 到 新 事件 模型 改 
变 的 方法 ， 下 面 的 例子 回 到 在 本 章 第 一 部 分 利用 事件 模型 来 证 明 的 一 些 
和 争议。 另外， 每 个 程序 包括 程序 片 和 应 用 程序 现在 都 可 以 借助 或 不 借助 
浏览 需 来 运行 。 





1. 文本 字段 
这 个 例子 同 TextField1.java 相 似 ， 但 它 增 加 了 显然 额外 的 行为 : 


//: TextNew.java 


// Text fields with Java 1.1 events 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
public class TextNew extends Applet { 
Button 
b1 = new Button("Get Text"), 


b2 = new Button("Set Text"); 


TextField 
t1 = new TextField(30), 
t2 = new TextField(30), 
t3 = new TextField(30); 

String s = new String(); 

public void init() { 
b1.addActionListener(new B1()); 
b2.addActionListener(new B2()); 
t1.addTextListener(new T1()); 
t1.addActionListener(new T1A()); 
t1.addKeyListener(new T1K()); 
add(b1); 
add(b2); 
add(t1); 
add(t2); 
add(t3); 

} 

class T1 implements TextListener { 
public void textValueChanged(TextEvent e) { 


t2.setText(t1.getText()); 


} 


class T1A implements ActionListener { 


private int count = 0; 
public void actionPerformed(ActionEvent e) { 


t3.setText("t1 Action Event " + count++); 


} 


class T1K extends KeyAdapter { 
public void keyTyped(KeyEvent e) { 
String ts = ti.getText(); 
if(e.getKeyChar() == 
KeyEvent.VK_BACK_SPACE) { 
// Ensure it's not empty: 
if( ts.length() > 0) { 
ts = ts.substring(0, ts.length() - 1); 


t1.setText(ts); 


} 


else 
t1.setText( 
t1.getText() + 
Character .toUpperCase( 
e.getKeyChar())); 
t1.setCaretPosition( 


ti1.getText().length()); 


// Stop regular character from appearing: 


e.consume(); 


} 


class B1 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
s = ti.getSelectedText(); 
if(s.length() == 0) s = ti.getText(); 


t1.setEditable(true); 


} 


class B2 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
ti1.setText("Inserted by Button 2: " + s); 


t1.setEditable(false); 


} 


public static void main(String[] args) { 
TextNew applet = new TextNew(); 
Frame aFrame = new Frame("TextNew"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 


public void windowClosing(WindowEvent e) 


System.exit(0); 
} 

}); 
aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 


} 
} ///:~ 


当 TextField t1 的 动作 接收 器 被 激活 时 ，TextField {3 就 是 一 个 需要 报告 的 
场所 。 我 们 注意 到 仪 当 我 们 按 下 “enter” 刍 时， 动作 接收 器 才 会 
为 “TextField” 所 激活 。 


TextField _ t1 附 有 几 个 接收 器 。T1 接 收 器 从 t 复 制 所 有 文字 到 包 ， 强 制 所 
有 字符 串 转 换 成 大 写 。 我 们 会 发 现 这 两 个 工作 同 是 进行 的 ， 并 且 如 果 我 
们 增加 T1K 接 收费 后 我 们 再 增加 T1 接 收 弗 ， 它 就 不 那么 重要 : 在 文字 字 
段 内 的 所 有 的 字符 串 将 一 直 被 强制 变 为 大 写 。 这 看 起 来 键盘 事件 一 直 在 
文字 组 件 事件 前 被 激活 ， 并 且 如 果 我 们 需要 保留 t2 的 字符 串 原来 输入 时 
的 样子 ， 我 们 就 必须 做 一 些 特别 的 工作 。 


T1K 有 着 其 它 的 一 些 有 趣 的 活动 。 我 们 必须 测试 backspace (因为 我 们 现 
在 控制 着 每 一 个 事件 ) 并 执行 删除 。caret 必 须 被 明确 地 设置 到 字段 的 结 
尾 ; 否则 它 不 会 像 我们 希望 的 运行 。 最 后 ， 为 了 防止 原来 的 字符 串 被 默 
认 的 机 制 所 处 理 ， 事 件 必 须 利用 为 事件 对 象 而 存在 的 consume() 方 法 

所 “ 耗 尽 ?。 这 会 通知 系统 停止 激活 其 余 特殊 事件 的 事件 处 理 嚣 。 


这 个 例子 同样 无 声 地 证 明了 设计 内 部 类 的 融 来 的 诸多 优点 。 注 意 下 面 的 


内 部 类 : 

















class T1 implements TextListener { 


public void textValueChanged(TextEvent e) { 


t2.setText(t1.getText()); 


tL 和 t2 不 属于 TI1 的 一 部 分 ， 并 且 到 目前 为 止 它们 都 是 很 容易 理解 的 ， 没 
有 任何 的 特殊 限制 。 这 是 因为 一 个 内 部 类 的 对 象 能 自动 地 捕捉 一 个 句柄 
到 外 部 的 创建 它 的 对 象 那里 ， 因 此 我 们 可 以 处 理 封装 类 对 象 的 方法 和 内 
容 。 正 像 我 们 看 到 的 ， 这 十 分 方便 (注释 @) 。 


©: 它 也 解决 了 “回调 ”的 问题 ， 不 必 为 Java 加 入 任何 令 人 恼火 的 “方法 指 
针 ” 特 性 。 

2. 文本 区 域 

Java 1.1 版 中 Text Area 最 重要 的 改变 就 深 动 条 。 对 于 TextArea 的 构建 器 而 
言 ， 我 们 可 以 立即 控制 TextArea 是 否 会 拥有 滚动 条 : KF, EAK, 
两 者 都 有 或 者 都 没有 。 这 个 例子 更 正 了 前 面 Java 1.0 版 TextAreal.java 程 
序 片 ， 演 示 了 Java 1.1 版 的 深 动 条 构建 器 : 


//: TextAreaNew. java 





// Controlling scrollbars with the TextArea 
// component in Java 1.1 

import java.awt.*; 

import java.awt.event.*; 

import java.applet.*; 


public class TextAreaNew extends Applet { 


Button bi = new Button("Text Area 1"); 
Button b2 = new Button("Text Area 2"); 
Button b3 = new Button("Replace Text"); 
Button b4 = new Button("Insert Text"); 
TextArea ti = new TextArea("ti", 1, 30); 


TextArea t2 


new TextArea("t2", 4, 30); 


TextArea t3 


new TextArea("t3", 1, 30, 
TextArea.SCROLLBARS_NONE); 
TextArea t4 = new TextArea("t4", 10, 10, 
TextArea.SCROLLBARS_VERTICAL_ONLY); 
TextArea t5 = new TextArea("t5", 4, 30, 
TextArea.SCROLLBARS_HORIZONTAL_ONLY) ; 
TextArea t6 = new TextArea("t6", 10, 10, 
TextArea.SCROLLBARS_BOTH) ; 
public void init() { 
b1i.addActionListener(new B1L()); 
add(b1); 
add(t1); 
b2.addActionListener(new B2L()); 
add(b2); 
add(t2); 
b3.addActionListener(new B3L()); 


add(b3) ; 


b4.addActionListener(new B4L()); 
add(b4); 
add(t3); add(t4); add(t5); add(té6); 
} 
class B1L implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


t5.append(ti.getText() + "\n"); 


} 


class B2L implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
t2.setText("Inserted by Button 2"); 
t2.append(": " + t1.getText()); 


t5.append(t2.getText() + "\n"); 


} 


class B3L implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
String s = " Replacement "; 


t2.replaceRange(s, 3, 3 + s.length()); 


} 


class B4L implements ActionListener { 


public void actionPerformed(ActionEvent e) { 


t2.insert(" Inserted ", 10); 


} 


public static void main(String[] args) { 
TextAreaNew applet = new TextAreaNew(); 
Frame aFrame = new Frame("TextAreaNew" ) ; 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


aFrame.add(applet, BorderLayout.CENTER) ; 
aFrame.setSize(300, 725); 

applet.init(); 

applet.start(); 

aFrame.setVisible(true); 


} 
LV] i 





我 们 发 现 只 能 在 构造 TextArea 时 能 够 控制 滚动 条 。 同 样 ， 即 使 TE_ AR 没 
我 们 滚动 光标 也 将 被 制止 〈 可 通过 运行 这 个 例子 中 验证 这 种 
行为 ) 。 


3. 复 选 框 和 单 选 钮 


正如 早先 指出 的 那样 ， 复 选 框 和 单 选 钮 都 是 同一 个 类 建立 的 。 单 选 钮 和 
复 选 框 略 有 不 同 ， 它 是 复 选 框 安置 到 CheckboxGroup 中 构成 的 。 在 其 中 
任 一 种 情况 下 ， 有 趣 的 IemEvent 事 件 为 我 们 创建 一 个 ItemListener 项 目 
接收 费 。 

当 处 理 一 组 复 选 框 或 者 单 选 钮 时 ， 我 们 有 一 个 不 错 的 选择 。 我 们 可 以 创 
建 一 个 新 的 内 部 类 去 为 每 个 复 选 框 处 理事 件 ， 或 者 创建 一 个 内 部 类 判断 
哪个 复 选 框 被 单 击 并 注册 一 个 内 部 类 单独 的 对 象 为 每 个 复 选 对 象 。 下 面 
的 例子 演示 了 两 种 方法 : 


//: RadioCheckNew. java 











// Radio buttons and Check Boxes in Java 1.1 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
public class RadioCheckNew extends Applet { 
TextField t = new TextField(30); 
Checkbox[] cb = { 
new Checkbox("Check Box 1"), 
new Checkbox("Check Box 2"), 
new Checkbox("Check Box 3") }; 
CheckboxGroup g = new CheckboxGroup(); 
Checkbox 
cb4 = new Checkbox("four", g, false), 


cb5 = new Checkbox("five", g, true), 


cb6 = new Checkbox("six", g, false); 
public void init() { 
t.setEditable(false); 
add(t); 
ILCheck il = new ILCheck(); 
for(int i = 0; i < cbh.length; i++) { 
cb[i].addItemListener(il); 
add(cb[i]); 
} 
cb4.addItemListener(new IL4()); 
cb5.addItemListener(new IL5()); 
cb6.addItemListener(new IL6()); 
add(cb4); add(cb5); add(chb6); 
} 
// Checking the source: 
class ILCheck implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 
for(int i = 0; i < cbh.length; i++) { 
if(e.getSource().equals(cb[i])) { 
t.setText("Check box " + (i + 1)); 


return; 


} 


// “VS. an individual class for each item: 
Class IL4 implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 


t.setText("Radio button four"); 


} 


class IL5 implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 


t.setText("Radio button five"); 


class IL6 implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 


t.setText("Radio button six"); 


} 

public static void main(String[] args) { 
RadioCheckNew applet = new RadioCheckNew(); 
Frame aFrame = new Frame("RadioCheckNew" ); 
aFrame.addWindowListener ( 


new WindowAdapter() { 


public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true) ; 
} 
} ///:~ 


ILCheck 拥 有 当 我 们 增加 或 者 减少 复 选 框 时 自动 调整 的 优点 。 当 然 ， 我 
们 对 单 选 钮 使 用 这 种 方法 也 同样 的 好 。 但 是 ， 它 仅 当 我 们 的 逻辑 足以 普 
遍 的 支持 这 种 方法 时 才 会 被 使 用 。 如 果 声 明 一 个 确定 的 信号 一 一 我 们 将 
重复 利用 独立 的 接收 器 类 ， 否 则 我 们 将 结束 一 串 条 件 语句 。 

4. 下 拉 列 表 


下 拉 列 表 在 Java 1.1 版 中 当 一 个 选择 被 改变 时 同样 使 用 ItemListener 去 告 
知 我 们 : 


//: ChoiceNew. java 





// Drop-down lists with Java 1.1 
import java.awt.*; 
import java.awt.event.*; 


import java.applet.*; 


public class ChoiceNew extends Applet { 
String[] description = { "Ebullient", "Obtuse", 
"Recalcitrant", "Brilliant", "Somnescent", 
"Timorous", "Florid", "Putrescent" }; 
TextField t = new TextField(100); 
Choice c = new Choice(); 
Button b = new Button("Add items"); 
int count = 0; 
public void init() { 
t.setEditable(false); 
for(int i = 0; i < 4; i++) 
c.addItem(description[count++]); 
add(t); 
add(c); 
add(b); 
c.addItemListener(new CL()); 
b.addActionListener(new BL()); 
} 
class CL implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 
t.setText("index: " + c.getSelectedIndex() 


+" " + e,toString()); 


} 


class BL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(count < description.length) 


c.addItem(description[count++] ) ， 


} 


public static void main(String[] args) { 
ChoiceNew applet = new ChoiceNew(); 
Frame aFrame = new Frame("ChoiceNew"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(750,100); 

applet.init(); 

applet.start(); 


aFrame.setVisible(true); 


de 


这 个 程序 中 没什么 特别 新 颖 的 东西 〈 除 了 Java 1.1 版 的 UI 类 里 少数 几 个 
值得 关注 的 缺陷 ) 。 


5. 列表 


我 们 消除 了 Java 1.0 中 List 设 计 的 一 个 缺陷 ， 就 是 List 不 能 像 我们 硕 望 的 
那样 工作 : 它 会 与 单 击 在 一 个 列表 元 素 上 发 生 冲 突 。 


//: ListNew.java 








// Java 1.1 Lists are easier to use 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
public class ListNew extends Applet { 
String[] flavors = { "Chocolate", "Strawberry", 
"Vanilla Fudge Swirl", "Mint Chip", 
"Mocha Almond Fudge", "Rum Raisin", 
"Praline Cream", "Mud Pie" }; 
// Show 6 items, allow multiple selection: 
List lst = new List(6, true); 
TextArea t = new TextArea(flavors.length, 30); 
Button b = new Button("test"); 
int count = 0; 


public void init() { 


t.setEditable(false); 
for(int i = 0; i < 4; i++) 
lst.addItem(flavors[count++]); 
add(t); 
add(lst); 
add(b); 
lst.addItemListener(new LL()); 
b.addActionListener(new BL()); 
} 
class LL implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 
t.setText(""); 
String[] items = lst.getSelectedItems(); 
for(int i = 0; i < items.length; i++) 


t.append(items[i] + "\n"); 


} 


class BL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(count < flavors.length) 


lst.addiItem(flavors[count++], 0); 


public static void main(String[] args) { 
ListNew applet = new ListNew(); 
Frame aFrame = new Frame("ListNew"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
} ///:~ 


我 们 可 以 注意 到 在 列表 项 中 无 需 特 别 的 逻辑 需要 去 文 持 一 个 单 击 动作 。 
我 们 正好 像 我 们 在 其 它 地 方 所 做 的 那样 附加 上 一 个 接收 费 。 


6. 菜单 


为 菜单 处 理事 件 看 起 来 受益 于 Java 1.1 版 的 事件 模型 ， 但 Java 生 成 菜单 的 
方法 党 第 麻烦 并 且 需 要 一 些 手 工 编写 代码 。 生 成 采 单 的 正确 方法 看 起 来 
像 资 源 而 不 是 一 些 代 码 。 请 牢 牢 记 住 编 程 工具 会 广泛 地 为 我 们 处 理 创建 
的 菜单 ， 因 此 这 可 以 减少 我 们 的 痛 百 (只 要 它们 会 同样 处 理 维护 任 

务 ! ) 。 男 外 ， 我 们 将 发 现 采 单 不 文 持 并 且 将 导致 混乱 的 事件 ， 亲 单项 




















使 用 ActionListeners 〈 动 作 接收 器 ) ， 但 复 选 框 薪 单项 使 用 
ItemListeners 〈 项 目 接 收 器 ) o SAL NTR APE RE Sc #FActionListeners 〈 动 
作 接 收 嚣 ) ， 但 通常 不 那么 有 用 。 一 般 来 说 ， 我 们 会 附加 接收 器 到 每 个 
荣 单 项 或 复 选 框 菜单 项 ， 但 下 面 的 例子 〈 对 先前 例子 的 修改 ) 演示 了 一 
个 联合 捕捉 多 个 菜单 组 件 到 一 个 单独 的 接收 右 类 的 方法 。 正 像 我 们 将 看 
到 的 ， 它 或 许 不 值得 为 这 而 激烈 地 争论。 


//: MenuNew. java 





// Menus in Java 1.1 
import java.awt.*; 
import java.awt.event.*; 
public class MenuNew extends Frame { 
String[] flavors = { "Chocolate", "Strawberry", 
"Vanilla Fudge Swirl", "Mint Chip", 
"Mocha Almond Fudge", "Rum Raisin", 
"Praline Cream", "Mud Pie" }; 
TextField t = new TextField("No flavor", 30); 
MenuBar mb1 = new MenuBar(); 
Menu f = new Menu("File"); 
Menu m = new Menu("Flavors"); 
Menu s = new Menu("Safety"); 
// Alternative approach: 
CheckboxMenuItem[] safety = { 
new CheckboxMenuItem("Guard"), 


new CheckboxMenuItem( "Hide" ) 


}; 
MenuItem[] file = { 
// No menu shortcut: 
new MenuItem("Open"), 
// Adding a menu shortcut is very simple: 
new MenuItem( "Exit", 
new MenuShortcut(KeyEvent.VK_E) ) 
}; 
// A second menu bar to swap to: 
MenuBar mb2 = new MenuBar(); 
Menu fooBar = new Menu("fooBar"); 
Menultem[] other = { 
new Menultem("Foo"), 
new Menultem("Bar"), 
new Menultem("Baz"), 
}; 
// Initialization code: 
{ 
ML ml = new ML(); 
CMIL cmil = new CMIL(); 
safety[0].setActionCommand("Guard"); 
safety[0].addItemListener(cmil) ; 


safety[1].setActionCommand("Hide"); 


safety[1].addItemListener(cmil) ; 
file[0].setActionCommand("Open"); 
file[0].addActionListener (ml) ; 
file[1].setActionCommand("Exit"); 
file[1].addActionListener(ml) ; 
other [0].addActionListener(new FooL()); 
other[1].addActionListener(new BarL()); 
other[2].addActionListener(new BazL()); 
} 
Button b = new Button("Swap Menus"); 
public MenuNew() { 
FL fl = new FL(); 
for(int i = 0; i < flavors.length; i++) { 
MenuItem mi = new MenulItem(flavors[i]); 
mi.addActionListener(fl); 
m.add(m1); 
// Add separators at intervals: 
if((i+1) % 3 == 0) 
m.addSeparator(); 
上 
for(int i = 0; i < safety.length; i++) 
s.add(safety[i]); 


f.add(s); 


for(int i = 0; i < file.length; i++) 
f.add(file[i]); 

mb1i.add(f); 

mb1.add(m); 

setMenuBar (mb1) ; 

t.setEditable(false); 

add(t, BorderLayout.CENTER) ; 

// Set up the system for swapping menus: 

b.addActionListener(new BL()); 

add(b, BorderLayout.NORTH) ; 

for(int i = 0; i < other.length; i++) 
fooBar.add(other[i]); 


mb2.add(fooBar ); 


class BL implements ActionListener { 


} 


public void actionPerformed(ActionEvent e) 
MenuBar m = getMenuBar(); 
if(m == mb1) setMenuBar(mb2); 


else if (m == mb2) setMenuBar(mb1); 


class ML implements ActionListener { 


public void actionPerformed(ActionEvent e) 


MenuItem target = (MenuItem)e.getSource(); 
String actionCommand = 
target.getActionCommand(); 
if(actionCommand.equals("Open")) { 
String s = t.getText(); 
boolean chosen = false; 
for(int 1 = 0; i < flavors.length; i++) 
if(s.equals(flavors[i])) chosen = true; 
if(!chosen) 
t.setText("Choose a flavor first!"); 
else 
t.setText("Opening "+ s +". Mmm, mm!"); 
} else if(actionCommand.equals("Exit")) { 
dispatchEvent ( 
new WindowEvent (MenuNew. this, 


WindowEvent .WINDOW_CLOSING) ) ; 


} 


class FL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
MenuItem target = (MenuItem)e.getSource(); 


t.setText(target.getLabel()); 


} 


// Alternatively, you can create a different 
// class for each different MenuItem. Then you 
// Don't have to figure out which one it is: 
Class FooL implements ActionListener { 

public void actionPerformed(ActionEvent e) { 


t.setText("Foo selected"); 


} 


class BarL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


t.setText("Bar selected"); 


J 


class BazL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


t.setText("Baz selected"); 


} 


class CMIL implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 


CheckboxMenuItem target = 


(CheckboxMenuItem)e.getSource(); 
String actionCommand = 

target .getActionCommand(); 
if(actionCommand.equals("Guard" ) ) 

t.setText("Guard the Ice Cream! " + 

"Guarding is " + target.getState()); 

else if(actionCommand.equals("Hide") ) 

t.setText("Hide the Ice Cream! " + 


"Is it cold? " + target.getState()); 


} 


public static void main(String[] args) { 
MenuNew f = new MenuNew(); 
f .addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


f.setSize(300, 200); 


f.setVisible(true); 


} ///:~ 


在 我 们 开始 初始 化 节 【( 由 注解 “Initialization ”code:” 后 的 右 大 括号 指明 ) 
的 前 面部 分 的 代码 同 先前 (Java ”1.0 版 ) 版 本 相同 。 这 里 我 们 可 以 注意 
到 项 目 接收 右 和 动作 接收 右 被 附加 在 不 同 的 菜单 组 件 上 。 


Java 1.1 文 持 “ 染 单 快 捷 键 >， 因 此 我 们 可 以 选择 一 个 荣 单 项 目 利 用 键盘 蔡 
代 鼠 标 。 这 十 分 的 简单 ;我 们 只 要 使 用 过 载 菜 单项 构建 器 设置 第 二 个 上 自 
变量 为 一 个 MenuShortcut 〈 沫 单 快 捷 键 事 件 ) OREN AY. SEAR GEREN 
建 句 设置 重要 的 方法 ， 当 它 按 下 时 不 可 思议 地 显示 在 沫 单项 上 上。 上面 的 
例子 增加 了 Control-E 到 “Exit” 


菜单 项 中 。 


我 们 同样 会 注意 setActionCommand0 的 使 用 。 这 看 似 一 点 陌生 因为 在 各 
种 情况 下 “action command” 完 全 同 菜单 组 件 上 的 标签 一 样 。 为 什么 不 正 
好 使 用 标签 代 蔡 可 选择 的 字符 串 昵 ? 这 个 难题 是 国际 化 的 。 如 果 我 们 重 
新 用 其 它 语言 写 这 个 程序 ， 我 们 只 需要 改变 菜单 中 的 标签 ， 并 不 审查 代 
码 中 可 能 包含 新 错误 的 所 有 逻辑 。 因 此 使 这 对 检查 文字 字符 串联 合 荣 单 
组 件 的 代码 而 言 变 得 简单 容易 ， 当 染 单 标签 能 改变 时 “动作 指令 ?可 以 不 
作 任 何 的 改变 。 所 有 这 些 代 码 同 “动作 指令 ”一同 工作 ， 因 此 它 不 会 受 改 
变 菜 单 标 签 的 有 影响。 注意 在 这 个 程序 中 ， 不 是 所 有 的 菜单 组 件 都 被 它们 
的 动作 指令 所 审查 ， 因 此 这 些 组 件 都 没有 它们 的 动作 指令 集 。 


大 多 数 的 构建 器 同 前 面 的 一 样 ， 将 几 个 调用 的 异常 增加 到 接收 器 中 。 大 
量 的 工作 发 生 在 接收 喜 里 。 在 前 面 例子 的 BEL 中 ， 某 单 交 普 发 生 。 在 ML 
H, “寻找 ring” 方 法 被 作为 动作 事件 〈ActionEvent) 的 资源 并 对 它 进 行 
造型 送 入 菜单 项 ， 然 后 得 到 动作 指令 字符 串 ， 再 通过 它 去 贯穿 串联 组 ， 
当然 条 件 是 对 它 进行 声 明 。 这 些 大 多 数 同 前 面 的 一 样 ， 但 请 注意 如 

果 “Exit” 被 选中 ， 通 过 进入 封装 类 对 象 的 句柄 〈MenuNew.this) 并 创建 
一 个 WINDOW_CLOSING 事 件 ， 一 个 新 的 窗口 事件 就 被 创建 了 。 新 的 
事件 被 分 配 到 封装 类 对 象 的 dispatchEvent() 方 法 ， 然 后 结束 调用 
windowsClosing0 内 部 帧 的 窗口 接收 器 (这 个 接收 器 作为 一 个 内 部 类 被 
创建 在 main0 里 ) ， 似 乎 这 是 “正常 ?产生 消息 的 方法 。 通 过 这 种 机 制 ， 
我 们 可 以 在 任何 情况 下 迅速 处 理 任何 的 信息 ， 因 此 ， 它 非常 的 强大 。 


FL 接收 器 是 很 简单 尽管 它 能 处 理 特殊 菜单 的 所 有 不 同 的 特色 。 如 果 我 们 
的 逻辑 十 分 的 简单 明了 ， 这 种 方法 对 我 们 束 很 有 用 处 ， 但 通常 ， 我 们 使 























用 这 种 方法 时 需要 与 FooL，BarL 和 BazL 一 道 使 用 ， 它 们 每 个 都 附加 到 
一 个 单独 的 菜单 组 件 上 ， 因 此 必然 无 需 训 试 逻 辑 ， 并 且 使 我 们 正确 地 辩 
识 出 谁 调用 了 接收 费 。 这 种 方法 产生 了 大 量 的 类 ， 内 部 代码 趋向 于 变 得 
小 巧 和 处 理 起 来 简单 、 安 全 。 


7. 对 话 框 


在 这 个 例子 里 直接 重 写 了 早期 的 ToeTest.java 程 序 。 在 这 个 新 的 版 本 里 ， 
任何 事件 都 被 安放 进 一 个 内 部 类 中 。 虽 然 这 完全 消除 了 需要 记录 产生 的 
任何 类 的 厂 烦 ， 作 为 ToeTest.java 的 一 个 例子 ， 它 能 使 内 部 类 的 概念 变 得 
不 那 东 远 。 在 这 点 ， 内 髓 类 被 散人 套 达 四 层 之 深 ! 我 们 需要 的 这 种 设计 决 
定 了 内 部 类 的 优点 是 否 值 得 增加 更 加 复杂 的 事物 。 男 外 ， 当 我 们 创建 一 
个 非 静 态 的 内 部 类 时 ， 我 们 将 捆绑 非 静 态 类 到 和 它 周 围 的 类 上 。 有 时 ， 单 
独 的 类 可 以 更 容易 地 被 复 用 。 


//: ToeTestNew. java 











// Demonstration of dialog boxes 
// and creating your own components 
import java.awt.*; 
import java.awt.event.*; 
public class ToeTestNew extends Frame { 
TextField rows = new TextField("3"); 
TextField cols = new TextField("3"); 
public ToeTestNew() { 
setTitle("Toe Test"); 
Panel p = new Panel(); 
p.setLayout(new GridLayout(2,2)); 


p.add(new Label("Rows", Label.CENTER)); 


p.add(rows); 
p.add(new Label("Columns", Label.CENTER) ); 
p.add(cols); 
add(p, BorderLayout.NORTH); 
Button b = new Button("go"); 
b.addActionListener(new BL()); 
add(b, BorderLayout.SOUTH) ; 
} 
static final int BLANK = 0; 
static final int XX = 1; 
static final int 00 = 2; 
class ToeDialog extends Dialog { 
// w = number of cells wide 
// h = number of cells high 
int turn = XX; // Start with x's turn 
public ToeDialog(int w, int h) { 
super (ToeTestNew. this, 

"The game itself", false); 
setLayout(new GridLayout(w, h)); 
for(int i = 0; i <w * h; i++) 

add(new ToeButton()); 
setSize(w * 50, h * 50); 


addwWindowListener(new WindowAdapter() { 


public void windowClosing(WindowEvent e){ 


dispose(); 


}); 


class ToeButton extends Canvas { 
int state = BLANK; 
ToeButton() { 
addMouseListener(new ML()); 


} 


public void paint(Graphics g) { 


int x1 O; 

int y1 = 0; 

int x2 = getSize().width - 1; 
int y2 = getSize().height - 1; 
g.drawRect(x1, y1, x2, y2); 

x1 = x2/4; 


yi = y2/4; 


int wide = x2/2; 


int high y2/2; 


if(state == XX) { 


g.drawLine(x1i, y1, 
x1 + wide, y1 + high); 
g.drawLine(x1, y1 + high, 
x1 + wide, y1); 
} 
if(state == 00) { 
g.drawOval(xi, yi, 


x1 + wide/2, y1 + high/2); 


} 


class ML extends MouseAdapter { 
public void mousePressed(MouseEvent e) { 

if(state == BLANK) { 
state = turn; 
turn = (turn == XX ? 00 : XX); 

} 

else 
state = (state == XX ? 00 : XX); 


repaint(); 


} 


class BL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
Dialog d = new ToeDialog( 
Integer.parseInt(rows.getText()), 
Integer.parseInt(cols.getText())); 


d.show(); 


} 


public static void main(String[] args) { 
Frame f = new ToeTestNew(); 
f .addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


f.setSize(200, 100); 


f.setVisible(true); 


} 
Sdn 


由 于 “静态 ”的 东西 只 能 位 于 类 的 外 部 一 级 ， 所 以 内 部 类 不 可 能 拥有 静态 


数据 或 者 静态 内 部 类 。 
8. 文件 对 话 框 
这 个 例子 是 直接 用 新 事件 模型 对 FileDialogTest.java 修 改 而 来 。 


//: FileDialogNew. java 


// Demonstration of File dialog boxes 
import java.awt.*; 
import java.awt.event.*; 
public class FileDialogNew extends Frame { 
TextField filename = new TextField(); 
TextField directory = new TextField(); 
Button open = new Button("Open"); 
Button save = new Button("Save"); 
public FileDialogNew() { 
setTitle("File Dialog Test"); 
Panel p = new Panel(); 
p.setLayout(new FlowLayout()); 
open.addActionListener (new OpenL()); 
p.add(open); 
save.addActionListener(new SaveL()); 
p.add(save); 
add(p, BorderLayout.SOUTH) ; 


directory.setEditable(false); 


filename.setEditable(false); 
p = new Panel(); 
p.setLayout(new GridLayout(2,1)); 
p.add( filename) ; 
p.add(directory); 
add(p, BorderLayout.NORTH); 
} 
class OpenL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
// Two arguments, defaults to open file: 
FileDialog d = new FileDialog( 
FileDialogNew.this, 
"What file do you want to open?"); 
d.setFile("*.java"); 
d.setDirectory("."); // Current directory 
d.show(); 
String yourFile = "*,.*"; 
if((yourFile = d.getFile()) != null) { 
filename.setText(yourFile) ; 
directory.setText(d.getDirectory()); 
} else { 
filename.setText("You pressed cancel"); 


directory.setText(""); 


} 


class SaveL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 

FileDialog d = new FileDialog( 
FileDialogNew.this, 
"What file do you want to save?", 
FileDialog.SAVE); 

d.setFile("*.java"); 

d.setDirectory("."); 

d.show(); 

String saveFile; 

if((saveFile = d.getFile()) != null) { 
filename.setText(saveFile) ; 
directory.setText(d.getDirectory()); 

} else { 
filename.setText("You pressed cancel"); 


directory.setText(""); 


} 


public static void main(String[] args) { 


Frame f = new FileDialogNew(); 
f .addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
f.setSize(250,110); 
f.setVisible(true); 
} 
} ///:~ 





如 末 所 有 的 改变 是 这 样 的 容易 那 将 有 多 棒 ， 但 至 少 它们 已 足够 容易 ， 并 
且 我 们 的 代码 已 受益 于 这 改进 的 可 读 性 上 。 


13.16.5 动态 绑 定 事件 

新 AWT 事 件 模型 给 我 们 带 来 的 一 个 好 处 就 是 灵活 性 。 在 老 的 模型 中 我 
们 被 迫 为 我 们 的 程序 动作 艰难 地 编写 代码 。 但 新 的 模型 我 们 可 以 用 单一 
方法 调用 增加 和 删除 事件 动作 。 下 面 的 例子 证 明了 这 一 点 : 


//: DynamicEvents.java 





// The new Java 1.1 event model allows you to 
// change event behavior dynamically. Also 
// demonstrates multiple actions for an event. 


import java.awt.*; 


import java.awt.event.*; 

import java.util.*; 

public class DynamicEvents extends Frame { 
Vector v = new Vector(); 
int i = 0; 
Button 


b1 


new Button("Button 1"), 


b2 


new Button("Button 2"); 

public DynamicEvents() { 
setLayout(new FlowLayout()); 
b1.addActionListener(new B()); 
b1.addActionListener(new B1()); 
b2.addActionListener(new B()); 
b2.addActionListener(new B2()); 
add(b1); 
add(b2); 

} 

class B implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


System.out.printin("A button was pressed"); 


} 


class CountListener implements ActionListener { 


int index; 

public CountListener(int i) { index = i; } 

public void actionPerformed(ActionEvent e) { 
System.out.printin( 


"Counted Listener " + index); 


J 
class B1 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
System.out.println("Button 1 pressed"); 
ActionListener a = new CountListener(i++); 
v.addElement(a); 


b2.addActionListener(a); 


} 
class B2 implements ActionListener { 
public void actionPerformed(ActionEvent e) { 

System.out.println("Button 2 pressed"); 

int end = v.size() -1; 

if(end >= 0) { 

b2.removeActionListener ( 
(ActionListener )v.elementAt(end)); 


v.removeElementAt(end); 


} 
public static void main(String[] args) { 
Frame f = new DynamicEvents(); 
f .addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
System.exit(0); 
} 
}); 
f.setSize(300, 200); 
f.show(); 
} 
} ///:~ 


这 个 例子 采取 的 新 手法 包括 : 


(1) 在 每 个 按钮 上 附着 不 少 于 一 个 的 接收 费 。 通 常 ， 组 件 把 事件 作为 多 
造型 处 理 ， 这 意味 着 我 们 可 以 为 单个 事件 注册 许多 接收 器 。 当 在 特殊 的 
组 件 中 一 个 事件 作为 单一 造型 被 处 理 时 ， 我 们 会 得 到 
TooManyListenersException 〈 即 太 多 接收 器 异常 ) 。 


(2) 程序 执行 期 间 ， 接 收 器 动态 地 被 从 按钮 B2 中 增加 和 删除 。 增 加 用 我 
们 前 面 见 到 过 的 方法 完成 ， 但 每 个 组 件 同样 有 一 个 removeXXXListener() 
RXXX EAE) 方法 来 删除 各 种 类 型 的 接收 器 。 


这 种 灵活 性 为 我 们 的 编程 提供 了 更 强大 的 能 








我 们 注意 到 事件 接收 顷 不 能 保证 在 命令 他 们 被 增加 时 可 被 调用 《虽然 事 
实 上 大 部 分 的 执行 工作 都 是 用 这 种 方法 完成 的 ) 。 


13.16.6 将 事务 逻辑 与 UI 逻辑 区 分 开 


一 般 而 言 ， 我 们 需要 设计 我 们 的 类 如 此 以 至 于 每 一 类 做 “一 件 事 ”。 当 涉 
及 用 户 接口 代码 时 就 更 显得 尤为 重要 ， 因 为 它 很 容易 地 封装 “您 要 做 什 
么 ”和 “怎样 显示 它 ”。 这 种 有 效 的 配合 防止 了 代码 的 重复 使 用 。 更 不 用 
说 它 令 人 满意 的 从 GUI 中 区 分 出 我 们 的 “事物 逻辑?。 使 用 这 种 方法 ， 我 
人 
GUI. 


FE Fe OEM RR FETE HY Te AS HLA BS RKB SR. DEER 
的 定位 规则 允许 所 有 新 事件 修改 后 立刻 生效 ， 并 且 这 是 如 此 一 个 引 人 注 
目的 设置 系统 的 方法 。 但 是 这 些 动作 对 象 可 以 被 在 一 些 不 同 的 应 用 程序 
使 用 并 且 因 此 不 会 被 一 些 特 殊 的 显示 模式 所 约束 。 它 们 会 合理 地 执行 动 
作 操 作 并 且 没 有 多 余 的 事件 。 

下 面 的 例子 演示 了 从 GUI 代码 中 多 么 地 轻松 的 区 分 事物 逻辑 : 


//: Separation.java 


























// Separating GUI logic and business objects 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
class BusinessLogic { 
private int modifier; 
BusinessLogic(int mod) { 


modifier = mod; 


public void setModifier(int mod) { 
modifier = mod; 

} 

public int getModifier() { 
return modifier; 

} 

// Some business operations: 

public int calculationi(int arg) { 

return arg * modifier; 

} 

public int calculation2(int arg) { 


return arg + modifier; 


} 


public class Separation extends Applet { 
TextField 
t = new TextField(20), 
mod = new TextField(20); 
BusinessLogic bl = new BusinessLogic(2); 
Button 


calc1 = new Button("Calculation 1"), 


calc2 new Button("Calculation 2"); 


public void init() { 


add(t); 
calci.addActionListener(new CalciL()); 
calc2.addActionListener(new Calc2L()); 
add(calci); add(calc2); 
mod.addTextListener(new ModL()); 
add(new Label("Modifier:")); 
add(mod); 

} 

static int getValue(TextField tf) { 
try { 

return Integer.parseInt(tf.getText()); 

} catch(NumberFormatException e) { 


return 0; 


} 


Class CalciL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
t.setText(Integer.toString/( 


b1l.calculationi(getValue(t)))); 


} 


class Calc2L implements ActionListener { 


public void actionPerformed(ActionEvent e) { 


t.setText(Integer.toString( 


bl.calculation2(getValue(t)))); 


} 


class ModL implements TextListener { 
public void textValueChanged(TextEvent e) { 


bl.setModifier(getValue(mod)); 


} 


public static void main(String[] args) { 
Separation applet = new Separation(); 
Frame aFrame = new Frame("Separation"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(200, 200); 

applet.init(); 

applet.start(); 


aFrame.setVisible(true); 


} 
} ///:~ 





可 以 看 到 ， 事 物 逻 辑 是 一 个 直接 完成 它 的 操作 而 不 需要 提示 并 且 可 以 在 
GUI 环 境 下 使 用 的 类 。 它 正 适 合 它 的 工作 。 区 分 动作 记录 了 所 有 UI 的 详 
细 资 料 ， 并 且 它 只 通过 它 的 公共 接口 与 事物 逻辑 交流 。 所 有 的 操作 围绕 
中 心 通过 UI 和 事物 逻辑 对 象 来 回 获 取信 息 。 因 此 区 分 ， 轮 流 做 它 的 工 
作 。 因 为 区 分 中 只 知道 它 同 事物 过 辑 对 象 对 话 〈 也 就 是 说 ， 它 没有 高 度 
的 结合 ) ， 它 可 以 被 强迫 同 其 它 类 型 的 对 象 对 话 而 没有 更 多 的 烦恼 。 


思考 从 事物 逻辑 中 区 分 UI 的 条 件 ， 同 样 思 考 当 我 们 调整 传统 的 Java 代 码 
使 它 运行 时 ， 怎 样 使 它 更 易 存活 。 


13.16.7 推荐 编码 方法 


内 部 类 是 新 的 事件 模型 ， 并 且 事实 上 旧 的 事件 模型 连同 新 库 的 特征 都 被 
它 好 的 文 持 ， 依 赖 老式 的 编程 方法 无 颖 增加 了 一 个 新 的 混乱 的 因素 。 现 
在 有 更 多 不 同 的 方法 为 我 们 编写 讨厌 的 代码 。 凑 巧 的 是 ， 这 种 代码 显现 
在 本 书 中 和 程序 样本 中 ， 并 且 甚 至 在 文件 和 程序 样本 中 同 SUN 公 司 区 别 
开 来 。 在 这 一 市 中 ， 我 们 将 看 到 一 些 关 于 我 们 会 和 不 会 运行 新 AWT 的 
争执 ， 并 由 加 我 们 展示 除了 可 以 原 谎 的 情况 ， 我 们 可 以 随时 使 用 接收 器 
类 去 解决 我 们 的 事件 处 理 需 要 来 结束 。 因 为 这 种 方法 同样 是 最 简单 和 最 
清晰 的 方法 ， 它 将 会 对 我 们 学 习 它 构成 有 效 的 帮助 。 


在 看 到 任何 事 以 前 ， 我 们 知道 尽管 Java ” 1.1 向 后 兼容 Java 1.0 (也 就 是 
说 ， 我 们 可 以 在 1.1 中 编译 和 运行 1.0 的 程序 ) ， 但 我 们 并 不 能 在 同一 个 
程序 里 混合 事件 模型 。 换 言 之 ， 当 我 们 试图 集成 老 的 代码 到 一 个 新 的 程 
序 中 时 ， 我 们 不 能 使 用 老式 的 action() 方 法 在 同一 个 程序 中 ， 因 此 我 们 必 
须 决 定 是 否 对 新 程序 使 用 老 的 ， 难 以 维护 的 方法 或 者 升级 老 的 代码 。 这 
不 会 有 太 多 的 竞争 因为 新 的 方法 对 老 的 方法 而 言 是 如 此 的 优秀 。 


1. 准则 : 运行 它 的 好 方法 


为 了 给 我 们 一 些 事物 来 进行 比较 ， 这 儿 有 一 个 程序 例子 演示 疝 我 们 推荐 
的 方法 。 到 现在 它 会 变 得 相当 的 熟悉 和 舒适 。 


//: GoodIdea,java 





























// The best way to design classes using the new 
// Java 1.1 event model: use an inner class for 
// each different event. This maximizes 

// flexibility and modularity. 

import java.awt.*; 

import java.awt.event.*; 

import java.util.*; 

public class GoodIdea extends Frame { 


Button 


b1 new Button("Button 1"), 


b2 


new Button("Button 2"); 

public GoodIdea() { 
setLayout(new FlowLayout()); 
b1.addActionListener(new B1L()); 
b2.addActionListener(new B2L()); 
add(b1); 
add(b2); 

} 

public class B1L implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


System.out.println("Button 1 pressed"); 


} 
public class B2L implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


System.out.println("Button 2 pressed"); 


} 
public static void main(String[] args) { 
Frame f = new GoodIdea(); 
f .addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
System.out.println( "Window Closing"); 
System.exit(0); 
} 
}); 
f.setSize(300, 200); 
f.setVisible(true); 
} 
} ///:~ 





这 是 磊 有 反 微 不 足 道 的 ,每 个 按钮 有 它 自 己 的 印 出 一 些 事物 到 控制 台 的 
接收 器 。 但 请 注意 在 整个 程序 中 这 不 是 一 个 条 件 语 句 ， 或 者 是 一 些 表 

示 “ 我 想 要 知道 怎样 使 事件 发 生 ” 的 语句 。 每 块 代码 都 与 运行 有 关 ， 而 不 
征 类 型 检验 。 也 就 是 说 ， 这 是 最 好 的 编写 我 们 的 代码 的 方法 ， 不 仅仅 是 
它 更 易 使 我 们 理解 概念 ， 至 少 是 使 我 们 更 易 阅 读 和 维护 。 剪 切 和 粘贴 到 








新 的 程序 是 同样 如 此 的 容易 。 

2. 将 主 类 作为 接收 顷 实 现 

第 一 个 坏 主 意 是 一 个 通 利 的 和 推荐 的 方法 。 这 使 得 主 类 〈 有 代表 性 的 是 
但 它 能 变 成 一 些 类 ) 执行 各 种 不 同 的 接收 右 。 下 面 是 一 个 
F: 


//: BadIdeai.java 


// Some literature recommends this approach, 
// but it's missing the point of the new event 
// model in Java 1.1 
import java.awt.*; 
import java.awt.event.*; 
import java.util.*; 
public class BadIdeai extends Frame 
implements ActionListener, WindowListener { 
Button 


b1 


new Button("Button 1"), 


b2 = new Button("Button 2"); 


public BadIdeai() { 
setLayout(new FlowLayout()); 
addWindowListener(this); 
b1.addActionListener(this); 
b2.addActionListener(this); 


add(b1); 


add(b2); 


i; 


public void actionPerformed(ActionEvent e) { 


Object source = e.getSource(); 


if (source 


== b1) 


System.out.println("Button 1 pressed"); 


else if(source == b2) 


System.out.println("Button 2 pressed"); 


else 


System.out.printin("Something else"); 


} 


public void windowClosing(WindowEvent e) { 


System.out.println("Window Closing"); 


System.exit(0); 


} 

public 
public 
public 
public 
public 
public 


public 


void 


void 


void 


void 


void 


void 


windowClosed(WindowEvent e) {} 
windowDeiconified(WindowEvent e) {} 
windowIconified(WindowEvent e) {} 
windowActivated(WindowEvent e) {} 
windowDeactivated(WindowEvent e) {} 


windowOpened(WindowEvent e) {} 


static void main(String[] args) { 


Frame f = new BadIdea1(); 


f.setSize(300, 200); 
f.setVisible(true); 


} 
A f/f 








这 样 做 的 用 途 显示 在 下 述 三 行 里 : 
addWindowListener(this); 
b1.addActionListener(this); 
b2.addActionListener(this); 


因为 Badideal 执 行动 作 接收 器 和 窗 中 接收 右 ， 这 些 程序 行当 然 可 以 接 
受 ， 并 且 如 果 我 们 一 直 坚 持 设 法 使 少量 的 类 去 减少 服务 器 检索 期 间 的 程 
序 片 载 入 的 作法 ， 它 看 起 来 变 成 一 个 不 错 的 主意 。 但 是 : 


(1) Java 1.1 版 文 持 JAR 文 件 ， 因 此 所 有 我 们 的 文件 可 以 被 放置 到 一 个 单 
一 的 压缩 的 JAR 文 件 中 ， 只 需要 一 次 服务 器 检索 。 我 们 不 再 需要 为 
Internet 效 率 而 减少 类 的 数量 。 


(2) ”上面 的 代码 的 组 件 更 加 的 少 ， 因 此 它 难 以 抓 住 和 烙 贴 。 注 意 我 们 必 
须 不 仅 要 执行 各 种 各 样 的 接口 为 我 们 的 主 类 ， 但 在 actionPerformed() 方 
法 中 ， 我 们 利用 一 串 条 件 语句 测试 哪个 动作 被 完成 了 。 不 仅仅 是 这 个 状 
态 倒退 ， 远 离 接收 器 模型 ， 除 此 之 外 ， 我 们 不 能 简单 地 重复 使 用 
actionPerformed() 方 法 因为 它 是 指定 为 这 个 特殊 的 应 用 程序 使 用 的 。 将 
这 个 程序 例子 与 GoodIdea.java 进 行 比 较 ， 我 们 可 以 正好 捕捉 一 个 接收 器 
类 并 粘贴 它 和 最 小 的 焦急 到 任何 地 方 。 另 外 我 们 可 以 为 一 个 单独 的 事件 











3. 方法 的 混合 
第 二 个 bad idea 混合 了 两 种 方法 : 使 用 内 髓 接收 费 类 ， 但 同样 执行 一 个 


或 更 多 的 接收 顷 接 口 以 作为 主 类 的 一 部 分 。 这 种 方法 无 需 在 书 中 和 文件 
中 进行 解释 ， 而 且 我 可 以 腾 测 到 Java 开 发 者 认为 他 们 必须 为 不 同 的 目的 





而 采取 不 同 的 方法 。 但 我 们 却 不 必 
倾 回 于 使 用 内 髓 接收 各 类 。 


//: BadIdea2.java 


在 我 们 编程 时 ， 我 们 或 许可 能 会 





// An improvement over BadIdeai.java, since it 
// uses the WindowAdapter as an inner class 
// instead of implementing all the methods of 
// WindowListener, but still misses the 
// Valuable modularity of inner classes 
import java.awt.*; 
import java.awt.event.*; 
import java.util.*; 
public class BadIdea2 extends Frame 
implements ActionListener { 
Button 
b1 = new Button("Button 1"), 
b2 = new Button("Button 2"); 
public BadIdea2() { 
setLayout(new FlowLayout()); 
addwWindowListener(new WL()); 
b1.addActionListener(this); 
b2.addActionListener(this); 


add(b1); 


add(b2); 
ae 
public void actionPerformed(ActionEvent e) { 
Object source = e.getSource(); 
if(source == b1) 
System.out.println("Button 1 pressed"); 
else if(source == b2) 
System.out.println( "Button 2 pressed"); 
else 
System.out.println( "Something else"); 
} 
class WL extends WindowAdapter { 
public void windowClosing(WindowEvent e) { 
System.out.println("Window Closing"); 


System.exit(0); 


} 


public static void main(String[] args) { 
Frame f = new BadIdea2(); 
f.setSize(300, 200); 


f.setVisible(true); 


} ///:~ 





为 actionPerformed() 动 作 完 成 方法 同 主 类 紧密 地 结合 ， 所 以 难以 复 用 
代码 。 它 的 代码 读 起 来 同样 是 凌乱 和 令 人 厌烦 的 ， 远 远 超过 了 内 部 类 方 
Aa 1.1 版 中 为 事件 使 用 那些 老 的 思 
路 。 

4. 继承 一 个 组 件 


创建 一 个 新 类 型 的 组 件 时 ， 在 运行 事件 的 老 方法 中 ， 我 们 会 经 向 看 到 不 
同 的 地 方 发 生 了 变化 。 这 里 有 一 个 程序 例子 来 演示 这 种 新 的 工作 方法 : 


//: GoodTechnique. java 





// Your first choice when overriding components 
// should be to install listeners. The code is 
// much safer, more modular and maintainable. 
import java.awt.*; 
import java.awt.event.*; 
class Display { 
public static final int 
EVENT = ©, COMPONENT = 1, 
MOUSE = 2, MOUSE_MOVE = 3, 
FOCUS = 4, KEY = 5, ACTION = 6, 
LAST = 7; 
public String[] evnt; 
Display() { 


evnt = new String[LAST]; 


for(int i = 0; i < LAST; i++) 
evnt[i] = new String(); 
} 
public void show(Graphics g) { 
for(int i = 0; i < LAST; i++) 


g.drawString(evnt[i], ©, 10 * i+ 10); 


} 


Class EnabledPanel extends Panel { 
Color c; 
int id; 
Display display = new Display(); 


public EnabledPanel(int i, Color mc) { 


setLayout(new BorderLayout()); 

add(new MyButton(), BorderLayout.SOUTH); 
addComponentListener(new CL()); 
addFocusListener(new FL()); 
addKeyListener(new KL()); 
addMouseListener(new ML()); 


addMouseMotionListener(new MML()); 


// To eliminate flicker: 
public void update(Graphics g) { 

paint(g); 

} 

public void paint(Graphics g) { 
g.setColor(c); 
Dimension s = getSize(); 
g.fillRect(0, ©, s.width, s.height); 
g.setColor(Color.black) ; 
display.show(g); 

} 

// Don't need to enable anything for this: 

public void processEvent(AwTEvent e) { 

display.evnt[Display.EVENT]= e.toString(); 
repaint(); 
super .processEvent(e); 

} 

class CL implements ComponentListener { 
public void componentMoved(ComponentEvent e){ 

display.evnt[Display.COMPONENT] = 
"Component moved"; 


repaint(); 


public void 
componentResized(ComponentEvent e) { 
display.evnt[Display.COMPONENT] = 
"Component resized"; 
repaint(); 
} 
public void 
componentHidden(ComponentEvent e) { 
display.evnt[Display.COMPONENT] = 
"Component hidden"; 
repaint(); 
} 
public void componentShown(ComponentEvent e){ 
display.evnt[Display.COMPONENT] = 
"Component shown"; 


repaint(); 


i 


class FL implements FocusListener { 
public void focusGained(FocusEvent e) { 
display.evnt[Display.FOCUS] = 
"FOCUS gained"; 


repaint(); 


} 


public void focusLost(FocusEvent e) { 
display.evnt[Display.FOCUS] = 
"FOCUS lost"; 


repaint(); 


} 


class KL implements KeyListener { 
public void keyPressed(KeyEvent e) { 
display.evnt[Display.KEY] = 
"KEY pressed: "; 
showCode(e); 
} 
public void keyReleased(KeyEvent e) { 
display.evnt[Display.KEY] = 
"KEY released: "; 
showCode(e); 
} 
public void keyTyped(KeyEvent e) { 
display.evnt[Display.KEY] = 
"KEY typed: "; 


showCode(e); 


void showCode(KeyEvent e) { 

int code = e.getKeyCode(); 
display.evnt[Display.KEY] += 
KeyEvent.getKeyText(code); 


repaint(); 


} 


class ML implements MouseListener { 
public void mouseClicked(MouseEvent e) { 
requestFocus(); // Get FOCUS on click 
display.evnt[Display.MOUSE] = 
"MOUSE clicked"; 
showMouse(e); 
} 
public void mousePressed(MouseEvent e) { 
display.evnt[Display.MOUSE] = 
"MOUSE pressed"; 
showMouse(e); 
} 
public void mouseReleased(MouseEvent e) { 
display.evnt[Display.MOUSE] = 
"MOUSE released"; 


showMouse(e); 


} 


public void mouseEntered(MouseEvent e) { 
display.evnt[Display.MOUSE] = 
"MOUSE entered"; 
showMouse(e); 
} 
public void mouseExited(MouseEvent e) { 
display.evnt[Display.MOUSE] = 
"MOUSE exited"; 
showMouse(e); 
} 
void showMouse(MouseEvent e) { 
display.evnt[Display.MOUSE] += 
", x=" + e.getx() + 
", y=" + e.getY(); 


repaint(); 


i 


class MML implements MouseMotionListener { 
public void mouseDragged(MouseEvent e) { 
display.evnt[Display.MOUSE_MOVE] = 
"MOUSE dragged"; 


showMouse(e); 


} 


public void mouseMoved(MouseEvent e) { 

display.evnt[Display.MOUSE_MOVE] = 
"MOUSE moved"; 

showMouse(e); 

} 

void showMouse(MouseEvent e) { 
display.evnt[Display.MOUSE_MOVE] += 

, X=" + e.getXx() + 

, y=" + e.getY(); 


repaint(); 


Í 
class MyButton extends Button { 
int clickCounter; 
String label = ""; 
public MyButton() { 
addActionListener(new AL()); 
} 
public void paint(Graphics g) { 
g.setColor(Color.green); 


Dimension s = getSize(); 


g.fillRect(0, ©, s.width, s.height); 
g.setColor(Color.black); 
g.drawRect(0, ©, s.width - 1, s.height - 1); 
drawLabel(g); 
} 
private void drawLabel(Graphics g) { 
FontMetrics fm = g.getFontMetrics(); 
int width = fm.stringWidth(label); 
int height = fm.getHeight(); 
int ascent = fm.getAscent(); 
int leading = fm.getLeading(); 
int horizMargin = 
(getSize().width - width)/2; 
int verMargin = 
(getSize().height - height)/2; 
g.setColor(Color.red); 
g.drawString(label, horizMargin, 
verMargin + ascent + leading); 
} 
class AL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
clickCounter++; 


label = "click #" + clickCounter + 


"" + e,toString(); 


repaint(); 


public class GoodTechnique extends Frame { 
GoodTechnique() { 
setLayout(new GridLayout(2,2)); 
add(new EnabledPanel(1, Color.cyan)); 
add(new EnabledPanel(2, Color.lightGray) ); 
add(new EnabledPanel(3, Color.yellow) ); 
} 
public static void main(String[] args) { 
Frame f = new GoodTechnique(); 
f.setTitle("Good Technique"); 
f .addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
System.out.printlin(e); 
System.out.println( "Window Closing"); 


System.exit(0); 


3); 
f.setSize(700, 700); 
f.setVisible(true); 

} 
} ///:~ 





这 个 程序 例子 同样 证 明了 各 种 各 样 的 发 现 和 显示 关于 它们 的 信息 的 事 
件 。 这 种 显示 是 一 种 集中 显示 信息 的 方法 。 一 组 字符 串 去 获取 关于 每 种 
类 型 的 事件 的 信息 ， 并 且 show0) 方 法 对 任何 图 像 对 象 都 设置 了 一 个 句 
铺 ， 我 们 采用 并 直接 地 写 在 外 现代 码 上 。 这 种 设计 是 有 总 的 被 六 事 伯 


激活 面板 代表 了 这 种 新 型 的 组 件 。 它 是 一 个 底部 有 一 个 按钮 的 彩色 的 面 
板 ， 并 且 它 由 利用 接收 器 类 为 每 一 个 单独 的 事件 来 引发 捕捉 所 有 发 生 在 
它 之 上 的 事件 ， 除 了 那些 在 激活 面板 过 载 的 老式 的 processEvent() 方 法 
(注意 它 应 该 同样 调用 super.processEvent()) 。 利 用 这 种 方法 的 唯一 理 
由 是 它 捕 捉 发 生 的 每 一 个 事件 ， 因 此 我 们 可 以 观察 持续 发 生 的 每 一 事 
件 。processEvent0 方 法 没有 更 多 的 展示 代表 每 个 事件 的 字符 串 ， 人 否则 它 
会 不 得 不 使 用 一 串 条 件 语句 去 寻找 事件 。 在 其 它 方面 ， 内 骸 接 收 类 早已 
清晰 地 知道 被 发 现 的 事件 。〔 假 定 我 们 注册 它们 到 组 件 ， 我 们 不 需要 任 
何 的 控件 的 逻辑 ， 这 将 成 为 我 们 的 目的 。〉 因 此 ， 它 们 不 会 去 检查 任何 
事件 ;这 些 事 件 正好 做 它们 的 原材料 。 

每 个 接收 器 修改 显示 字符 串 和 它 的 指定 事件 ， 并 且 调 用 重 画 方法 
a i 0 
WR JN? : 














public void update(Graphics g) { 
paint(g); 
} 


我 们 不 会 始终 需要 过 载 update0， 但 如 条 我 们 写 下 一 些 闪 烁 的 程序 ， 并 
运行 它 。 默 认 的 最 新 版 本 的 清除 背景 然后 调用 paint0) 方 法 重新 夯 出 一 些 





图 画 。 这 个 消除 动作 通常 会 产生 内 炮 ， 但 是 不 必要 的 ， 因 为 paint0 重 男 
了 整个 的 外 观 。 


我 们 可 以 看 到 许多 的 接收 需 一 一 但 是 ， 对 接收 需 和 输入 检查 指令 ， 但 我 们 
却 不 能 接收 任何 组 件 不 文 持 的 事件 。《〈 不 像 BadTechnuque.java 那 样 我 们 
能 时 时 刻 刻 看 到 ) 。 


试验 这 个 程序 是 十 分 的 有 教育 意义 的 ， 因 为 我 们 学 习 了 许多 的 关于 在 
Java 中 事件 发 生 的 方法 。 一 则 它 展示 了 大 多 数 开 窗 口 的 系统 中 设计 上 的 
IG: 它 相 当 的 难以 去 单 击 和 释放 鼠标 ， 除 非 移 动 它 ， 并 且 当 我 们 实际 
上 正 试 图 用 鼠标 单 击 在 某 物 体 上 时 开 窗 口 的 会 常常 认为 我 们 是 在 拖 动 。 
一 个 解决 这 个 问题 的 方案 是 使 用 mousePressed0) 女 标 按 下 方法 和 

Re sees 鼠标 释放 方法 去 代 和 着 mouseClicked0 鼠 标 单 击 方法 ， 然 
后 判断 是 否 去 调用 我 们 目 己 的 以 时 间 和 4 个 像素 的 鼠标 六 后 作用 

的 “mouseReallyClickedO 真 实 的 鼠标 单 击 ”方法 。 


5. 览 脚 的 组 件 继 承 


男 一 种 做 法 是 调用 enableEvent(O 方 法 ， 并 将 与 希望 控制 的 事件 对 应 的 模 

型 传递 给 它 《〈 许 多 参考 书 中 都 曾 提 及 这 种 做 法 ) 。 这 样 做 会 造成 那些 事 

件 被 发 送 至 老式 方法 (尽管 它们 对 Java 1. 1 来 说 是 新 的 ) ， 并 采用 象 

的 名 字 。 也 必须 要 记 住 调用 基础 类 版 本 。 下 面 
它 看 起 来 的 样子 。 


//: BadTechnique. java 








// It's possible to override components this way, 
// but the listener approach is much better, so 
// why would you? 

import java.awt.*; 

import java.awt.event.*; 

class Display { 


public static final int 


EVENT = 0, COMPONENT = 1, 
MOUSE = 2, MOUSE_MOVE = 3, 
FOCUS = 4, KEY = 5, ACTION = 6, 
LAST = 7; 


public String[] evnt; 
Display() { 
evnt = new String[LAST]; 


©; i < LAST; i++) 


for(int i 
evnt[i] = new String(); 
} 
public void show(Graphics g) { 
for(int i = 0; i < LAST; i++) 


g.drawString(evnt[i], ©, 10 * i+ 10); 


} 


class EnabledPanel extends Panel { 
Color c; 
int id; 
Display display = new Display(); 


public EnabledPanel(int i, Color mc) { 


setLayout(new BorderLayout()); 


add(new MyButton(), BorderLayout.SOUTH) ; 
// Type checking is lost. You can enable and 
// process events that the component doesn't 
// capture: 
enableEvents( 
// Panel doesn't handle these: 
AWTEvent.ACTION_EVENT_MASK | 
AWTEvent .ADJUSTMENT_EVENT_MASK | 
AWTEvent.ITEM_EVENT_MASK | 
AWTEvent.TEXT_EVENT_MASK | 
AWTEvent .WINDOW_EVENT_MASK | 
// Panel can handle these: 
AWTEvent ..COMPONENT_EVENT_MASK | 
AWTEvent.FOCUS_EVENT_MASK | 
AWTEvent.KEY_EVENT_MASK | 
AWTEvent .MOUSE_EVENT_MASK_ | 
AWTEvent .MOUSE_MOTION_EVENT_MASK | 
AWTEvent .CONTAINER_EVENT_MASK) ; 
// You can enable an event without 
// overriding its process method. 
} 
// To eliminate flicker: 


public void update(Graphics g) { 


paint(g); 
} 
public void paint(Graphics g) { 
g.setColor(c); 
Dimension s = getSize(); 
g.fillRect(0, ©, s.width, s.height); 
g.setColor(Color.black); 
display.show(g); 
} 
public void processEvent(AwTEvent e) { 
display.evnt[Display.EVENT]= e.toString(); 
repaint(); 
super .processEvent(e); 
} 
public void 
processComponentEvent(ComponentEvent e) { 
switch(e.getID()) { 
case ComponentEvent .COMPONENT_MOVED: 
display.evnt[Display.COMPONENT] = 
"Component moved"; 
break; 
case ComponentEvent.COMPONENT_RESIZED: 


display.evnt[Display.COMPONENT] = 


"Component resized"; 
break; 
case ComponentEvent .COMPONENT_HIDDEN: 
display.evnt[Display.COMPONENT] = 
"Component hidden"; 
break; 
case ComponentEvent .COMPONENT_SHOWN: 
display.evnt[Display.COMPONENT] = 
"Component shown"; 
break; 
default: 
} 
repaint(); 
// Must always remember to call the "super" 
// version of whatever you override: 
super .processComponentEvent(e); 
} 
public void processFocusEvent(FocusEvent e) { 
switch(e.getID()) { 
case FocusEvent.FOCUS_GAINED: 
display.evnt[Display.FOCUS] = 
"FOCUS gained"; 


break; 


case FocusEvent.FOCUS_LOST: 
display.evnt[Display.FOCUS] = 
"FOCUS lost"; 
break; 
default: 
} 
repaint(); 
super .processFocusEvent(e)j; 
} 
public void processKeyEvent(KeyEvent e) 
switch(e.getID()) { 


case KeyEvent.KEY_PRESSED: 


display.evnt[Display.KEY] 
"KEY pressed: "; 
break; 


case KeyEvent.KEY_RELEASED: 


display.evnt[Display.KEY] 
"KEY released: "; 
break; 


case KeyEvent.KEY_TYPED: 


display.evnt[Display.KEY] 
"KEY typed: "; 


break; 


default: 
} 
int code = e.getKeyCode(); 
display.evnt[Display.KEY] += 
KeyEvent.getKeyText(code) ; 
repaint(); 
super .processKeyEvent(e); 
} 
public void processMouseEvent (MouseEvent e) 
switch(e.getID()) { 
case MouseEvent .MOUSE_CLICKED: 


requestFocus(); // Get FOCUS on click 


display.evnt [Display .MOUSE ] 
"MOUSE clicked"; 
break; 


case MouseEvent .MOUSE_PRESSED: 


display.evnt [Display .MOUSE ] 
"MOUSE pressed"; 
break; 
case MouseEvent .MOUSE_RELEASED: 
display.evnt[Display.MOUSE] = 
"MOUSE released"; 


break; 


case MouseEvent .MOUSE_ENTERED : 


display.evnt [Display .MOUSE ] 
"MOUSE entered"; 
break; 


case MouseEvent .MOUSE_EXITED: 


display.evnt [Display .MOUSE ] 
"MOUSE exited"; 
break; 
default: 


} 


display.evnt[Display.MOUSE] += 


x 
| 


= " + e.getX() + 


= " + e.getY(); 


< 
| 


repaint(); 

super .processMouseEvent(e); 
} 
public void 
processMouseMotionEvent(MouseEvent e) { 

switch(e.getID()) { 

case MouseEvent .MOUSE_DRAGGED: 
display.evnt[Display.MOUSE_MOVE] = 
"MOUSE dragged"; 


break; 


case MouseEvent .MOUSE_MOVED : 
display.evnt[Display.MOUSE_MOVE] = 
"MOUSE moved"; 
break; 
default: 
} 
display.evnt[Display.MOUSE_MOVE] += 
", x=" + e.getX() + 


= " + e.getY(); 


< 
| 


repaint(); 


super .processMouseMotionEvent(e); 


} 


class MyButton extends Button { 

int clickCounter; 

String label = ""; 

public MyButton() { 
enableEvents(AWTEvent .ACTION_EVENT_MASK) ; 

} 

public void paint(Graphics g) { 
g.setColor(Color.green); 
Dimension s = getSize(); 


g.fillRect(0, ©, s.width, s.height); 


g.setColor(Color.black) ; 
g.drawRect(0, 0, s.width - 1, s.height - 1); 
drawLabel(g); 
} 
private void drawLabel(Graphics g) { 
FontMetrics fm = g.getFontMetrics(); 
int width = fm.stringWidth(label); 
int height = fm.getHeight(); 
int ascent = fm.getAscent(); 
int leading = fm.getLeading(); 
int horizMargin = 
(getSize().width - width)/2; 
int verMargin = 
(getSize().height - height)/2; 
g.setColor(Color.red); 
g.drawString(label, horizMargin, 
verMargin + ascent + leading); 
} 
public void processActionEvent(ActionEvent e) { 
clickCounter++; 
label = "click #" + clickCounter + 
"" + e,toString(); 


repaint(); 


super.processActionEvent(e); 


public class BadTechnique extends Frame { 
BadTechnique() { 
setLayout(new GridLayout(2,2)); 
add(new EnabledPanel(1, Color.cyan)); 
add(new EnabledPanel(2, Color.lightGray) ); 
add(new EnabledPanel(3, Color.yellow) ); 
// You can also do it for Windows: 
enableEvents(AWTEvent .WINDOW_EVENT_MASK) ; 
} 
public void processWindowEvent(WindowEvent e) { 
System.out.printlin(e); 
if(e.getID() == WindowEvent .WINDOW_CLOSING) { 
System.out.println( "Window Closing"); 


System.exit(0); 


} 


public static void main(String[] args) { 
Frame f = new BadTechnique(); 


f.setTitle("Bad Technique"); 


f.setSize(700, 700); 
f.setVisible(true); 


} 
} ///:~ 


的 确 ， 它 能 够 工作 。 但 却 实在 太 览 脚 ， 而 且 很 难 编号、 了 阅读、 调试 、 维 
护 以 及 再 生 。 既 然 如 此 ， 为 什么 还 不 使 用 内 部 接收 器 类 呢 ? 


13.17 Java 1.1 用 户 接口 API 


Java ”1.1 版 同样 增加 了 一 些 重 要 的 新 功能 ， 包 括 焦点 过 历 ， 桌 面色 彩 访 
问 ， 打 印 “ 沙 箱 内 ”及 早期 的 竞 贴 板 支 持 。 





我 们 想 在 一 个 鼠标 单 击 上 捕捉 键盘 焦点 ， 我 们 可 以 捕捉 忌 标 按 下 事件 并 
且 调 用 requestFocus0O 需 求 焦 点 方法 。 


13.17.1 桌面 颜色 


利用 桌面 若 色 ， 我 们 可 知道 当前 用 户 果 面 都 有 哪些 颜色 选择 。 这 样 一 
来 ， 就 可 在 必要 的 时 候 通过 自己 的 程序 来 运用 那些 颜色 。 颜 色 都 会 得 以 
自动 初始 化 ， 并 置 于 SystemColor 的 static 成 员 中 ， 所 以 要 做 的 唯一 事情 
就 是 读 取 目 己 感 兴趣 的 成 员 。 各 种 名 字 的 意义 是 不 言 而 喻 的 : desktop, 
activeCaption, activeCaptionText, activeCaptionBorder, 
inactiveCaption, inactiveCaptionText, inactiveCaptionBorder, 
window,  windowBorder, windowText, menu, menuText, text, 
textText, textHighlight, textHighlightText, textInactiveText, control, 
controlText, controlHighlight, controlLtHighlight, controlShadow, 
controlDkShadow, scrollbar, info 〈 用 于 帮助 ) 以 及 infoText (HF ËS 
助 文 字 ) 。 














13.17.2 打印 


非常 不 幸 ， 打 印 时 没有 多 少 事 情 是 可 以 自动 进行 的 。 相 反 ， 为 完成 打 
印 ， 我 们 必须 经 历 大 量 机 械 的 、 非 OO《〈 面 问 对 象 ) 的 步骤 。 但 打印 一 
个 图 形 化 的 组 件 时 ， 可 能 多 少 有 点 儿 上 自动 化 的 意思 : 默认 情况 下 ， 
print() 方 法 会 调用 paint() 来 完成 自己 的 工作 。 大 多 数 时 候 这 都 已 经 足够 
了 ， 但 假如 还 想 做 一 些 特别 的 事情 ， 残 必须 知道 页 面 的 几何 尺寸 。 


下 和 面 这 个 例子 同时 演示 了 文字 和 图 形 的 打印 ， 以 及 打印 图 形 时 可 以 采取 
的 不 同方 法 。 此 外 ， 它 也 对 打印 支持 进行 了 测试 : 


//: PrintDemo.java 














// Printing with Java 1.1 

import java.awt.*; 

import java.awt.event.*; 

public class PrintDemo extends Frame { 
Button 


printText = new Button("Print Text"), 


printGraphics = new Button("Print Graphics"); 


TextField ringNum = new TextField(3); 
Choice faces = new Choice(); 


Graphics g = null; 


Plot plot = new Plot3(); // Try different plots 


Toolkit tk = Toolkit.getDefaultToolkit(); 
public PrintDemo() { 
ringNum.setText("3"); 
ringNum.addTextListener(new RingL()); 
Panel p = new Panel(); 
p.setLayout(new FlowLayout()); 
printText.addActionListener(new TBL()); 
p.add(printText); 
p.add(new Label("Font:")); 
p.add(faces); 


printGraphics.addActionListener(new GBL()); 


p.add(printGraphics); 
p.add(new Label("Rings:")); 
p.add(ringNum) ; 
setLayout(new BorderLayout()); 
add(p, BorderLayout.NORTH); 
add(plot, BorderLayout.CENTER) ; 
String[] fontList = tk.getFontList(); 
for(int i = 0; i < fontList.length; i++) 
faces.add(fontList[i]); 
faces.select("Serif"); 
} 
class PrintData { 
public PrintJob pj; 
public int pageWidth, pageHeight; 
PrintData(String jobName) { 
pj = getToolkit().getPrintJob( 
PrintDemo.this, jobName, null); 
if(pj != null) { 
pagewidth = pj.getPageDimension().width; 
pageHeight= pj.getPageDimension().height; 


g = pj.getGraphics(); 


void end() { pj.end(); } 
} 
class ChangeFont { 
private int stringHeight; 
ChangeFont(String face, int style,int point) { 
if(g != null) { 
g.setFont(new Font(face, style, point)); 
stringHeight = 


g.getFontMetrics().getHeight(); 


} 
int stringWidth(String s) { 
return g.getFontMetrics().stringwidth(s); 
} 
int stringHeight() { return stringHeight; } 
} 
class TBL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
PrintData pd = 
new PrintData("Print Text Test"); 
// Null means print job canceled: 
if(pd == null) return; 


String s = "PrintDemo"; 


ChangeFont cf = new ChangeFont( 
faces.getSelectedItem(), Font.ITALIC, 72); 
g.drawString(s, 
(pd.pageWidth - cf.stringWidth(s)) / 2, 
(pd.pageHeight - cf.stringHeight()) / 3); 
s = "A smaller point size"; 
cf = new ChangeFont( 
faces.getSelectedItem(), Font.BOLD, 48); 
g.drawString(s, 
(pd.pagewidth - cf.stringWidth(s)) / 2, 
(int)((pd.pageHeight - 
cf.stringHeight())/1.5)); 
g.dispose(); 


pd.end(); 


} 
class GBL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
PrintData pd = 
new PrintData("Print Graphics Test"); 
if(pd == null) return; 
plot.print(g); 


g.dispose(); 


pd.end(); 


} 


class RingL implements TextListener { 
public void textValueChanged(TextEvent e) { 
int i = 1; 
try { 
i = Integer.parseInt(ringNum.getText()); 


} catch(NumberFormatException ex) { 


plot.rings = i; 


plot.repaint(); 


i 


public static void main(String[] args) { 
Frame pdemo = new PrintDemo(); 
pdemo.setTitle("Print Demo"); 
pdemo.addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


}); 


pdemo.setSize(500, 500); 


pdemo.setVisible(true); 


} 


class Plot extends Canvas { 
public int rings = 3; 
} 
class Plot1 extends Plot { 
// Default print() calls paint(): 
public void paint(Graphics g) { 
int w = getSize().width; 
int h = getSize().height; 
int xc =w / 2; 
int yc = w/ 2; 


int x = 0, y 


lI 
© 


for(int i = 0; i < rings; i++) 
if(x < xc && y < yc) { 
g.drawOval(x, y, w, h); 
x += 10; y += 10; 


w -= 20; h -= 20; 


} 


class Plot2 extends Plot { 

// To fit the picture to the page, you must 
// know whether you're printing or painting: 
public void paint(Graphics g) { 

int w, h; 
if(g instanceof PrintGraphics) { 
PrintJob pj = 


((PrintGraphics)g).getPrintJob(); 


w = pj.getPageDimension().width; 
h = pj.getPageDimension().height; 
} 
else { 
w = getSize().width; 
h = getSize().height; 
} 


int xc =w / 2; 

int yc =w / 2; 

int x = 0, y = 9; 

for(int i = 0; i < rings; i++) { 
if(x < xc && y < yc) { 


g.drawOval(x, y, w, h); 


x += 10; y += 10; 


w -= 20; h -= 20; 


} 


class Plot3 extends Plot { 

// Somewhat better. Separate 
// printing from painting: 
public void print(Graphics g) { 

// Assume it's a PrintGraphics object: 
PrintJob pj = 
((PrintGraphics)g).getPrintJob(); 

int w = pj.getPageDimension().width; 
int h = pj.getPageDimension().height; 
doGraphics(g, w, h); 

} 

public void paint(Graphics g) { 
int w = getSize().width; 


int h 


getSize().height; 
doGraphics(g, w, h); 
} 


private void doGraphics( 


Graphics g, int w, int h) { 
int xc =w / 2; 
int yc = w/ 2; 
int x = 0, y=0; 
for(int i = 0; i < rings; i++) { 
1f(x < xc && y < yc) { 
g.drawOval(x, y, w, h); 
x += 10; y += 10; 


w -= 20; h -= 20; 


} 
Y Ffir 





这 个 程序 允许 我 们 从 一 个 选择 列表 框 中 选择 字体 (并 且 我 们 会 注意 到 很 
多 有 用 的 字体 在 Java 1.1 版 中 一 直 受 到 严格 的 限制 ， 我 们 没有 任何 可 以 
利用 的 优秀 字体 安装 在 我 们 的 机 器 上 ) 。 它 使 用 这 些 字 体 去 打出 粗 体 ， 
笠 体 和 不 同 大 小 的 文字 。 另 外 ， 一 个 新 型 组 件 调用 过 的 绘图 被 创建 ， 以 
用 来 示范 网 形 。 当 打印 网 形 时 ， 绘 图 拥有 的 ring 将 显示 在 屏幕 上 和 打印 
在 纸 上 ， 并 且 这 三 个 衍生 类 Plotl，Plot2，Plot3 用 不 同 的 方法 去 完成 任 
务 以 便 我 们 可 以 看 到 我 们 选择 的 事物 。 同 样 ， 我 们 也 能 在 一 个 绘图 中 改 
变 一 些 ring 一 一 这 很 有 趣 ， 因 为 它 证 明了 Java 1.1 版 中 打印 的 脆弱 。 在 我 
的 系统 里 ， 当 ring 计 数 显示 “too high”( 究 况 这 是 什么 意思 ? ) 时 ， 打 印 
机 给 出 错误 信息 并 且 不 能 正确 地 工作 ， 而 当 计 数 给 出 “low enough” 信 息 
时 ， 打 印 机 又 能 工作 得 很 好 。 我 们 也 会 注意 到 ， 当 打印 到 看 起 来 实际 大 
小 不 相符 的 纸 时 页 面 的 大 小 便 产 生 了 。 这 些 特点 可 能 被 装 入 到 将 来 发 行 
的 Java 中 ， 我 们 可 以 使 用 这 个 程序 来 测试 它 。 


这 个 程序 为 促进 重复 使 用 ， 不 论 何 时 都 可 以 封装 功能 到 内 部 类 中 。 例 

















如 ， 不 论 何 时 我 想 开始 打印 工作 “〈 不 论 图 形 或 文字 ) ， 我 必须 创建 一 个 
PrintJob 打 印 工作 对 象 ， 该 对 象 拥有 它 目 己 的 连同 页 面 宽度 和 高 度 的 图 
形 对 象 。 创 建 的 PrintJob 打 印 工 作对 象 和 提取 的 页 面 尺寸 一 起 被 封装 进 
PrintData class 打 印 类 中 。 


1. 打印 文字 


打印 文字 的 概念 简单 明了 : 我 们 选择 一 种 字体 和 大 小 ， 决 定 字 符 串 在 页 
面 上 存在 的 位 置 ， 并 且 使 用 Graphics.drawSrting(0) 方 法 在 页 面 上 画 出 字符 
串 就 行 了 。 这 意味 着 ， 不 管 怎样 我 们 必须 精确 地 计算 每 行 字 符 串 在 页 面 
上 存在 的 位 置 并 确定 字符 串 不 会 超出 页 面 底部 或 者 同 其它 行 冲突 。 如 果 
我 们 想 进 行 字 处 理 ， 我 们 将 进行 的 工作 与 我 们 很 相配 。ChangeFont 封 闭 
进 少 量 从 一 种 字体 到 其 它 的 字体 的 变更 方法 并 自动 地 创建 一 个 新 字体 对 
象 和 我 们 想 要 的 字体 ， 款 式 ( 粗 体 和 和 斜体 一 一 目前 还 不 支持 下 划 线 、 空 
心 等 ) 以 及 点 阵 大 小 。 它 同样 会 简单 地 计算 字符 串 的 宽度 和 高 度 。 当 我 
们 按 下 “Print text”* 按 钮 时 ，TBL 接 收 避 被 激活 。 我 们 可 以 注意 到 它 通 过 
反复 创建 ChangeFont 对 象 和 调用 drawString(0) 来 在 计算 出 的 位 置 打印 出 字 
ee ane 。 我 使 用 的 版 本 没有 出 
Bo ) 


2. 打印 图 形 


按 下 “Print graphics” 投 钮 时 ，GBL 接 收 器 会 被 激活 。 我 们 需要 打印 时 ， 
创建 的 PrintData 对 象 初始 化 ， 然 后 我 们 简单 地 为 这 个 组 件 调用 print0 打 
印 方法 。 为 强制 打印 ， 我 们 必须 为 图 形 对 象 调用 dispose() 处 理 方法 ， 并 
日 为 PrintData 对 象 调用 end0) 结 束 方法 (或 改变 为 为 PrintJob 调 用 end() 结 
束 方法 。) 


这 种 工作 在 绘图 对 象 中 继续 。 我 们 可 以 看 到 基础 类 绘图 是 很 简单 的 一 一 
它 扩展 画布 并 且 包 括 一 个 中 断 调用 ring 来 指明 多 少 个 集中 的 ring 需 要 画 
在 这 个 特殊 的 画布 上 。 这 三 个 衍生 类 展示 了 可 达到 一 个 目的 的 不 同 的 方 
法 : 国 在 屏幕 上 和 打印 的 页 面 上 。 


Plot1 采 用 最 简单 的 编程 方法 : 忽略 绘画 和 打印 的 不 同 ， 并 且 过 载 paint() 
绘画 方法 。 使 用 这 种 工作 方法 的 原因 是 默认 的 printO0 打 印 方法 简单 地 改 
变 工作 方法 转 而 调用 Paint()。 但 是 ， 我 们 会 注意 到 输出 的 尺寸 依赖 于 屏 
磋 上 男 布 的 大 小 ， 因 为 宽度 和 高 上 度 都 是 在 调用 Canvas.getSize() 方 法 时 决 
定 是 ， 所 以 这 是 合理 的 。 如 果 我 们 图 像 的 尺寸 一 值 都 是 固定 不 变 的 ， 其 














它 的 情况 都 可 接受 。 当 画 出 的 外 观 的 大 小 如 此 的 重要 时 ， 我 们 必须 深入 
了 解 的 尺寸 大 小 的 重要 性 。 不 凑巧 的 是 ， 就 像 我们 将 在 Plot2 中 看 到 的 一 
样 ， 这 种 方法 变 得 很 坏 手 。 因 为 一 些 我 们 不 知道 的 好 的 理由 ， 我 们 不 能 
简单 地 要 求 图 形 对 象 以 它 自己 的 大 小 画 出 外 观 。 这 将 使 整个 的 处 理工 作 
变 得 十 分 的 优良 。 相 反 ， 如 果 我 们 打印 而 不 是 绘画 ， 我 们 必须 利用 
RTTI instanceof 关 键 字 【〈 在 本 书 11 章 中 有 相应 描述 ) 来 测试 
PrintGrapics， 然 后 下 漳 造 型 并 调用 这 独特 的 PrintGraphics 方 法 : 
getPrintJob(0) 方 法 。 现 在 我 们 拥有 PrintJob 的 句柄 并 且 我 们 可 以 发 现 纸张 
的 高 度 和 宽度 。 这 是 一 种 hacky 的 方法 ， 但 也 许 这 对 它 来 说 是 合理 的 理 
由 。〔 在 其 它 方面 ， 到 如 今 我 们 看 到 一 些 其 它 的 库 设 计 ， 因 此 ， 我 们 可 
能 会 得 到 设计 者 们 的 想法 。) 


我 们 可 以 注意 到 Plot2 中 的 paintO 绘 画 方法 对 打印 和 绘图 的 可 能 性 进行 审 
查 。 但 是 因为 当 打 印 时 Print( 方 法 将 被 调用 ， 那 么 为 什么 不 使 用 那 种 方 
法 呢 ? 这 种 方法 同样 也 在 Plot3 中 也 被 使 用 ， 并 且 它 消除 了 对 instanceof 
使 用 的 需求 ， 因 为 在 Print() 方 法 中 我 们 可 以 假设 我 们 能 对 一 个 
PrintGraphics 对 象 造型 。 这 样 也 不 坏 。 这 种 情况 被 放置 公共 绘画 代码 到 
一 个 分 离 的 doGraphics() 方 法 的 办 法 所 改进 。 


2. 在 程序 片 内 运行 帧 


如 果 我 们 想 在 一 个 程序 片 中 打印 会 怎 以 样 呢 ? 很 好 ， 为 了 打印 任何 事物 
我 们 必须 通过 工具 组 件 对 象 的 getPrintJob() 方 法 拥有 一 个 PrintJob 对 象 ， 
设置 唯一 的 一 个 帧 对 象 而 不 是 一 个 程序 厂 对 象 。 于 是 它 似 乎 可 能 从 一 个 
应 用 程序 中 打印 ， 而 不 是 从 一 个 程序 片 中 打印 。 但 是 ， 它 变 为 我 们 可 以 
从 一 个 程序 片 中 创建 一 个 帜 (相反 的 到 目前 为 止 ， 我 在 程序 片 或 应 用 程 
序 例子 中 所 做 的 ， 都 可 以 生成 程序 片 并 安放 帧 。) 。 这 是 一 个 很 有 用 的 
技术 ， 因 为 它 允 许 我 们 在 程序 片 中 使 用 一 些 应 用 程序 (只 要 它们 不 妨碍 
程序 片 的 安全 ) 。 但 是 ， 当 应 用 程序 窗口 在 程序 片 中 出 现时 ， 我 们 会 注 
意 到 WEB 浏 览 器 插入 一 些 警 告 在 它 上 面 ， 其 中 一 些 产 生 “Warning:Applet 
Window. (275: 程序 片 窗口 ) ”的 字样 。 


我 们 会 看 到 这 种 技术 十 分 直接 的 安放 一 个 帧 到 程序 片 中 。 唯 一 的 事 是 当 
用 户 关 财 它 时 我 们 必须 增加 帧 的 代码 《〈 代 蔡 调 用 System.exitO ) : 


//: PrintDemoApplet. java 





























// Creating a Frame from within an Applet 
import java.applet.*; 
import java.awt.*; 
import java.awt.event.*; 
public class PrintDemoApplet extends Applet { 
public void init() { 
Button b = new Button("Run PrintDemo"); 
b.addActionListener(new PDL()); 
add(b); 
} 
class PDL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
final PrintDemo pd = new PrintDemo(); 
pd.addwindowListener(new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
pd.dispose(); 
} 
}); 
pd.setSize(500, 500); 


pd.show(); 


} ///:~ 





伴随 Java 1.1 版 的 打印 支持 功能 而 来 的 是 一 些 混乱 。 一 些 宣传 似乎 声明 
我 们 能 在 一 个 程序 片 中 打印 。 但 Java 的 安全 系统 包含 了 一 个 特点 ， 可 停 
i= 个 正在 初始 化 打印 工作 的 程序 片 ， 初始 化 程序 片 需要 通过 一 个 Web 
浏览 器 或 程序 片 浏览 器 来 进行 。 在 写作 这 本 书 时 ， 这 看 起 来 像 留 下 了 一 
个 未 定 的 争议 。 当 我 在 WEB 浏 览 器 中 运行 这 个 程序 时 ，printdemo( 打 
印 样本 ) 窗口 正好 出 现 ， 但 它 却 根本 不 能 从 浏览 器 中 打印 。 


13.17.3 剪贴 板 


Java 1.1 对 系统 剪贴 板 提 供 有 限 的 操作 文 持 〈 在 Java.awt.datatransfer 
package 里 ) 。 我 们 可 以 将 字符 串 作 这 文字 对 象 复 制 到 剪贴 板 中 ， 并 且 
我 们 可 以 从 让 前 贴 板 中 粘贴 文字 到 字符 中 对 角 中 。 当然 ， 剪 贴 板 被 设计 来 
容纳 各 种 类 型 的 数据 ， 存 在 于 剪贴 板 上 的 数据 通过 程序 运行 剪 切 和 粘贴 
进入 到 程序 中 。 虽 然 剪 切 板 目前 只 文 持 字 符 串 数据 ，Java 的 甬 切 板 API 
通过 “特色 ”概念 提供 了 展 好 的 可 扩展 性 。 当 数据 从 剪贴 板 中 出 来 时 ， 它 
拥有 一 个 相关 的 特色 集 ， 这 个 特色 集 可 以 被 修改 〈 例 如 ， 一 个 图 形 可 以 
被 表示 成 一 些 字符 串 或 者 一 幅 图 像 ) 并 且 我 们 会 注意 到 如 果 特 殊 的 剪贴 
板 数 据 支 持 这 种 特色 ， 我 们 会 对 此 十 分 的 感 兴趣 


下 面 的 程序 简单 地 对 TextArea 中 的 字符 串 数 据 进 行 剪 切 ， 复 制 ， 粘 贴 的 
操作 做 了 示范 。 我 们 将 注意 到 的 是 我 们 需要 按照 剪 切 、 复 制 和 粘贴 的 顺 
序 进 行 工 作 。 但 如 果 我 们 看 见 一 些 其 它 程 序 中 的 TextField 或 者 
TextArea， 我 们 会 发 现 它们 同样 也 目 动 地 文 持 剪 贴 板 的 操作 顺序 。 程 序 
中 简单 地 增加 了 剪贴 板 的 程序 化 控制 ， 如 果 我 们 想 用 它 来 捕捉 剪贴 板 上 
的 文字 到 一 些 非 文字 组 件 中 就 可 以 使 用 这 种 技术 。 


//: CutAndPaste.java 





























// Using the clipboard from Java 1.1 
import java.awt.*; 

import java.awt.event.*; 

import java.awt.datatransfer.*; 


public class CutAndPaste extends Frame { 


MenuBar mb = new MenuBar(); 
Menu edit = new Menu("Edit"); 
MenuItem 
cut = new MenuItem("Cut"), 
copy = new MenuItem("Copy"), 
paste = new MenuItem("Paste") ; 
TextArea text = new TextArea(20, 20); 
Clipboard clipbd = 
getToolkit().getSystemClipboard(); 
public CutAndPaste() { 
cut.addActionListener(new CutL()); 
copy.addActionListener (new CopyL()); 
paste.addActionListener(new PasteL()); 
edit.add(cut); 
edit.add(copy); 
edit.add(paste); 
mb.add(edit); 
setMenuBar (mb); 
add(text, BorderLayout.CENTER) ; 
} 
class CopyL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


String selection = text.getSelectedText(); 


StringSelection clipString = 
new StringSelection(selection) ; 


clipbd.setContents(clipString, clipString); 


} 


class CutL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
String selection = text.getSelectedText(); 
StringSelection clipString = 
new StringSelection(selection) ; 
clipbd.setContents(clipString, clipString); 
text.replaceRange("", 
text.getSelectionStart(), 


text.getSelectionEnd()); 


} 


Class PasteL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
Transferable clipData = 
clipbd.getContents(CutAndPaste.this); 
try { 
String clipString = 


(String)clipData. 


getTransferData( 
DataFlavor.stringFlavor); 
text.replaceRange(clipString, 
text.getSelectionStart(), 
text.getSelectionEnd()); 
} catch(Exception ex) { 


System.out.println("not String flavor"); 


} 
public static void main(String[] args) { 
CutAndPaste cp = new CutAndPaste(); 
cp.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) 
System.exit(0); 
} 
}); 


cp.setSize(300, 200); 


cp.setVisible(true); 


Eii 


创建 和 增加 沫 单 及 TextArea 到 如 今 似 乎 已 变 成 一 种 单调 的 活动 。 这 与 通 
过 工具 组 件 创建 的 毅 贴 板 字 段 dipbd 有 很 大 的 区 别 。 


所 有 的 动作 都 安置 在 接收 器 中 。CopyL 和 Cupl 接 收 器 同样 除了 最 后 的 
CutL 线 以 外 删除 被 复制 的 线 。 特 殊 的 两 条 线 是 StringSelection 对 象 从 字 
符 串 从 创建 并 调用 StringSelection 的 setContents0) 方 法 。 说 得 更 准确 些 ， 
就 是 放 一 个 字符 串 到 副 切 板 上 。 


在 PasteL 中 ， 数 据 被 剪贴 板 利用 getContents0) 进 行 分 解 。 任 何 返回 的 对 象 
都 是 可 移动 的 匿名 的 ， 并 且 我 们 并 不 真正 地 知道 它 里 面包 含 了 什么 。 有 
一 种 发 现 的 方法 是 调用 getTransferDateFlavorsO0， 返 回 一 个 DataFlavor 对 
象 数 组 ， 表 明 特 殊 对 象 文 持 这 种 特点 。 我 们 同样 能 要 求 它 通过 我 们 感 兴 
趣 的 特点 直接 地 使 用 IsDataFlavorSupported()。 但 是 在 这 里 使 用 一 种 大 胆 
的 方法 : 调用 getTransferData( ) 方法 ， 假 设 里 面 的 内 容 支 持 字 符 串 特 
色 ， 并 且 它 不 是 个 被 分 类 在 异常 处 理 器 中 的 难题 。 


在 将 来 ， 我 们 希望 更 多 的 数据 特色 能 够 被 支持 。 





13.18 可 视 编程 和 Beans 


运 今 为 止 ， 我 们 已 看 到 Java 对 创建 可 重复 使 用 的 代码 片 工作 而 言 是 多 么 
的 有 价值 。“ 最 大 限度 地 可 重复 使 用 ”的 代码 单元 拥有 类 ， 因 为 它 包 含 一 
个 紧密 结合 在 一 起 的 单元 特性 字段， 和 单元 动作 (方法 ) ， 它 们 可 以 
直接 经 过 混合 或 通过 继承 被 重复 使 用 。 


继承 和 多 形态 性 是 面向 对 象 编程 的 精华 ， 但 在 大 多 数 情 况 下 当 我 们 创建 
一 个 应 用 程序 时 ， 我 们 真正 最 想 要 的 恰恰 是 我 们 最 需要 的 组 件 。 我 们 希 
望 在 我 们 的 设计 中 设置 这 些 部 件 就 像 电子 工程 师 在 电路 板 上 创造 集成 电 
路 块 一 样 〈 在 使 用 Java 的 情况 下 ， 就 是 放 到 WEB 页 面 上 ) 。 这 似乎 会 成 
为 加 快 这 种 “模块 集合 ”编制 程序 方法 的 发 展 。 


“可 视 化 编程 ”最 早 的 成 功 非常 的 成 功 一 一 要 归功 于 微软 公司 的 
Visual Basic (VB， 可 视 化 Basic 语 言 ) ， 接 下 来 的 第 二 代 是 Borland 公 司 
Delphi (一 种 客户 /服务 器 数据 库 应 用 程序 开发 工具 ， 也 是 Java Beans 设 
计 的 主要 灵感 ) 。 这 些 编程 工具 的 组 件 的 像 征 就 是 可 视 化 ， 这 是 不 容 置 
疑 的 ， 因 为 它们 通常 展示 一 些 类 型 的 可 视 化 组 件 ， 例 如 :一 个 按 惯 或 一 
个 TextField。 事 实 上 ， 可 视 化 通常 表现 为 组 件 可 以 非常 精确 地 访问 运行 
中 程序 。 因 此 可 视 化 编程 方法 的 一 部 分 包含 从 一 个 调 色 盘 从 拖 放 一 个 组 
件 并 将 它 放 置 到 我 们 的 窗 体 中 。 应 用 程序 创建 工具 像 我 们 所 做 的 一 样 编 
写 程序 代码 ， 该 代码 将 导致 正在 运行 的 程序 中 的 组 件 被 创建 。 


简单 地 拖 放 组 件 到 一 个 帘 体 中 通常 不 足以 构成 一 个 完整 的 程序 。 一 般 情 
况 下 ， 我 们 需要 改变 组 件 的 特性 ， 例 如 组 件 的 色彩 ， 组 件 的 文字 ， 组 件 
连结 的 数据 库 ， 等 等 。 特 性 可 以 参照 属性 在 编程 时 进行 修改 。 我 们 可 以 
在 应 用 程序 构建 工具 中 巧妙 处 置 我 们 组 件 的 属性 ， 并 且 当 我 们 创建 程序 
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到 如 今 ， 我 们 可 能 习惯 于 使 用 对 象 的 多 个 特性 ， 这 也 是 一 个 动作 集合 。 
在 设计 时 ， 可 视 化 组 件 的 动作 可 由 事件 部 分 地 代表 ， 意 味 厦 “任何 事件 
都 可 以 发 生 在 组 件 上 ”。 通 常 ， 由 我 们 决定 想 发 生 的 事件 ， 当 一 个 事件 
发 生 时 ， 对 所 发 生 的 事件 连接 代码 。 


这 是 关键 性 的 部 分 : 应 用 程序 构建 工具 可 以 动态 地 询问 组 件 ( 利 用 映 

















象 ) 以 发 现 组 件 支 持 的 事件 和 属 件 。 一 旦 它 知 道 它们 的 状态 ， 应 用 程序 
构建 工具 就 可 以 显示 组 件 的 属性 并 允许 我 们 修改 它们 的 属性 〈 当 我 们 构 
建 程序 时 ， 保 存 它们 的 状态 ) ， 并 且 也 显示 这 些 事件 。 一 般 而 言 ， 我 们 
做 一 些 事件 像 双击 一 个 事件 以 及 应 用 程序 构建 工具 创建 一 个 代码 并 连接 
到 事件 上 。 当 事件 发 生 时 ， 我 们 不 得 不 编写 执行 代码 。 应 用 程序 构建 工 
具 累 计 为 我 们 做 了 大 量 的 工作 。 结 果 我 们 可 以 注意 到 程序 看 起 来 像 它 所 
假定 的 那样 运行 ， 并 且 依 赖 应 用 程序 构建 工具 去 为 我 们 管理 连接 的 详细 
资料 。 可 视 化 的 编程 工具 如 此 成 功 的 原因 是 它们 明显 加 快 构建 的 应 用 程 
序 的 处 理 过 程 一 一 当然 ， 用 户 接口 作为 应 用 程序 的 一 部 分 同样 的 好 。 


13.18.1 什么 是 Bean 


在 经 细节 处 理 后 ， 一 个 组 件 在 类 中 被 独特 的 具体 化 ， 真 正 地 成 为 一 块 代 
码 。 关 键 的 争议 在 于 应 用 程序 构建 工具 发 现 组 件 的 属性 和 事件 能 力 。 为 
了 创建 一 个 VB 组 件 ， 程 序 开 发 者 不 得 不 编写 正确 的 同时 也 是 复杂 烦琐 
的 代码 片 ， 接 下 来 由 某 些 协 议 去 展现 它们 的 事件 和 属性 。Delphi 是 第 二 
代 的 可 视 化 编程 工具 并 且 这 种 开发 语言 主动 地 围绕 可 视 化 编程 来 设计 因 
此 它 更 容易 去 创建 一 个 可 视 化 组 件 。 但 是 ，Java 带 来 了 可 视 化 的 创作 组 
件 做 为 Java Beans 最 高 级 的 “装备 "”， 因 为 一 个 Bean 就 是 一 个 类 。 我 们 不 
从 再 为 制造 任 何 的 Bean 而 编号 一 些 特 殊 的 代码 或 者 使 用 特殊 的 编程 说 
。 事 实 上 ， 我 们 唯一 需要 做 的 是 略微 地 修改 我 们 对 我 们 方法 命名 的 办 
a a 通知 应 用 程序 构建 工具 是 否 是 一 个 属性 ， 一 个 事件 或 是 一 个 
wae 


在 Java 的 文件 中 ， 命 名 规则 被 错误 地 曲解 为 “设计 范式 "。 这 十 分 的 不 
幸 ， 因 为 设计 范式 〈 参 见 第 16 章 ) 车 来 不 少 的 麻烦 。 命 名 规则 不 是 设计 
范式 ， 它 是 相当 的 简单 


(1) ”因为 属性 被 命名 为 xxx， 我 们 代表 性 的 创建 两 个 方法 : getXxx() 和 
setXxx()。 注 意 get 或 set 后 的 第 一 个 字母 小 写 以 产生 属性 

名 。“get” 和 “set” 方 法 产生 同样 类 型 的 自 变 量 。“set*” 和 “get” 的 属性 名 和 
类 型 名 之 间 没 有 关系 。 


(2) 对 于 布尔 逻辑 型 属性 ， 我 们 可 以 使 用 上 面 的 “get* 和 “set”* 方 法 ， 但 我 
们 也 可 以 用 “is” 代 替 “ get”. 


(3) Bean 的 普通 方法 不 适合 上 面 的 命名 规则 ， 但 它们 是 公用 的 。 
































4. 对 于 事件 ， 我 们 使 用 "listener〈 接 收 器 ) "方法 。 这 种 方法 完全 同 我 们 
看 到 过 的 方法 相同 : (addFooBarListener(FooBarListener) 和 
removeFooBarListener(FooBarListener) 方 法 用 来 处 理 FooBar 事 件 。 大 多 
数 时 候 内 建 的 事件 和 接收 器 会 满足 我 们 的 需要 ， 但 我 们 可 以 创建 自己 的 
事件 和 接收 器 接口 。 


上 面 的 第 一 点 回答 了 一 个 关于 我 们 可 能 注意 到 的 从 Java 1.0 到 Java 1.1 的 
改变 的 问题 ， 一 些 方法 的 名 字 太 过 于 短小 ， 显 然 改 写 名 字 守 无 意义 。 现 
在 我 们 可 以 看 到 为 了 制造 Bean 中 的 特殊 的 组 件 ， 大 多 数 的 这 些 修改 不 得 
不 适合 于 “get” 和 “set”* 命 名 规则 。 

现在 ， 我 们 已 经 可 以 利用 上 面 的 这 些 指导 方针 去 创建 一 个 简单 的 Bean: 


//: Frog.java 











// A trivial Java Bean 
package frogbean; 
import java.awt.*; 
import java.awt.event.*; 
class Spots {} 
public class Frog { 
private int jumps; 
private Color color; 
private Spots spots; 
private boolean jmpr; 
public int getJumps() { return jumps; } 
public void setJumps(int newJumps) { 


jumps = newJumps; 


public Color getColor() { return color; } 

public void setColor(Color newColor) { 
color = newColor; 

} 

public Spots getSpots() { return spots; } 

public void setSpots(Spots newSpots) { 
spots = newSpots,; 

} 

public boolean isJumper() { return jmpr; } 

public void setJumper(boolean j) { jmpr = j; } 

public void addActionListener ( 

ActionListener 1) { 

Po 

} 

public void removeActionListener ( 

ActionListener 1) { 

Tf aaa 

} 

public void addKeyListener(KeyListener 1) { 
Fl ies 

} 

public void removeKeyListener(KeyListener 1) { 


LA is 


} 
// An "ordinary" public method: 
public void croak() { 


System.out.println("Ribbet!"); 


} 
} ///:~ 


首先 ， 我 们 可 看 到 Bean 束 是 一 个 类 。 通 常 ， 所 有 我 们 的 字段 会 被 作为 专 
用 ， 并 且 可 以 接近 的 唯一 办 法 是 通过 方法 。 紧 接着 的 是 命名 规则 ， 属 性 
是 jump，color，jumper，spots (注意 这 些 修改 是 在 第 一 个 字母 在 属性 名 
的 情况 下 进行 的 ) 。 虽 然 内 部 确定 的 名 字 同 最 早 的 三 个 例子 的 属性 名 一 
样 ， 在 jumper 中 我 们 可 以 看 到 属性 名 不 会 强迫 我 们 使 用 任何 特殊 的 内 部 
可 变 的 名 字 《 或 者 ， 真 的 拥有 一 些 内 部 的 可 变 的 属性 名 ) 。 


Bean 事 件 的 句柄 是 ActionEvent 和 KeyEvent， 这 是 根据 有 关 接 收 器 

的 “add” 和 “remove” 命 名 方法 得 出 的 。 最 后 我 们 可 以 注意 到 普通 的 方法 
croak() 一 直 是 Bean 的 一 部 分 ， 仪 仪 是 因为 它 是 一 个 公共 的 方法 ， 而 不 是 
因为 它 符 合 一 些 命名 规则 。 


13.18.2 用 Introspector 提 取 BeanInfo 


当 我 们 拖 放 一 个 Bean 的 调 色 板 并 将 它 放 入 到 窗 体 中 时 ， 一 个 Bean 的 最 关 
键 的 部 分 的 规则 发 生 了 。 应 用 程序 构建 工具 必须 可 以 创建 Bean (如果 它 
是 默认 的 构建 句 的 话 ， 它 就 可 以 做 ) 然后 ， 在 此 范围 外 访问 Bean 的 源 代 
码 ， 提 取 所 有 的 必要 的 信息 以 创立 属性 表 和 事件 处 理 占 。 


解决 方案 的 一 部 分 在 11 章 结尾 部 分 已 经 显现 出 来 : Java” ”1.1 版 的 映 象 允 
许 一 个 匿名 类 的 所 有 方法 被 发 现 。 这 完美 地 解决 了 Bean 的 难题 而 无 需 我 
们 使 用 一 些 特殊 的 语言 关键 字 像 在 其 它 的 可 视 化 编程 语言 中 所 需要 的 那 
样 。 事 实 上 ， 一 个 主要 的 原因 是 映 象 增加 到 Java 1.1 版 中 以 支持 
Beans (尽管 映 象 同样 支持 对 象 串联 和 远程 方法 调用 ) 。 因 为 我 们 可 能 
希望 应 用 程序 构建 工具 的 开发 者 将 不 得 不 映 象 每 个 Bean 并 且 通 过 它们 的 
方法 搜索 以 找到 Bean 的 属性 和 事件 。 
































这 当然 是 可 能 的 ， 但 是 Java 的 研制 者 们 和 希望 为 每 个 使 用 它 的 用 户 提 供 一 
个 标准 的 接口 ， 而 不 仅仅 是 使 Bean 更 为 简单 易 用 ， 不 过 他 们 也 同样 提供 
了 一 个 创建 更 复杂 的 Bean 的 标准 方法 。 这 个 接口 束 是 Introspector 类 ， 在 
这 个 类 中 最 重要 的 方法 静态 的 getBeanInfo()。 我 们 通过 一 个 类 处 理 这 个 
方法 并 且 getBeanInfo(0) 方 法 全 面 地 对 类 进行 查询 ， 返 回 一 个 我 们 可 以 进 
行 详细 研究 以 发 现 其 属性 、 方 法 和 事件 的 BeanInfo 对 象 。 


通常 我 们 不 会 留意 这 样 的 一 些 事物 一 一 我 们 可 能 会 使 用 我 们 大 多 数 的 现 
成 的 Bean， 并 且 我 们 不 需要 了 解 所 有 的 在 底层 运行 的 技术 细节 。 我 们 会 
简单 地 拖 放 我 们 的 Bean 到 我 们 窗 体 中 ， 然 后 配置 它们 的 属性 并 且 为 事件 
编写 处 理 器 。 无 论 如 何 它 都 是 一 个 有 趣 的 并 且 是 有 教育 意义 的 使 用 
Introspector 来 显示 关于 Bean 信 息 的 练习 ， 好 啦 ， 朵 话 少 说 ， 这 里 有 一 个 
工具 请 运行 它 “〈 我 们 可 以 在 forgbean 子 目录 中 找到 它 ) : 


//: BeanDumper.java 




















// A method to introspect a Bean 
import java.beans.*; 
import java.lang.reflect.*; 
public class BeanDumper { 
public static void dump(Class bean) { 
BeanInfo bi = null; 
try { 
bi = Introspector.getBeanInfo( 
bean, java.lang.Object.class); 
} catch(IntrospectionException ex) { 
System.out.printin("Couldn't introspect " + 
bean.getName()); 


System.exit(1); 


} 


PropertyDescriptor[] properties = 
bi.getPropertyDescriptors(); 
for(int i = 0; i < properties.length; i++) { 


Class p = properties[i].getPropertyType(); 


System.out.printin( 
"Property type:\n " + p.getName()); 
System. out.printin( 
"Property name:\n " + 
properties[i].getName()); 
Method readMethod = 
properties[i].getReadMethod(); 
if(readMethod != null) 
System.out.printin( 
"Read method:\n " + 
readMethod.toString()); 
Method writeMethod = 
properties[i].getWriteMethod(); 
if(writeMethod != null) 
System.out.printin( 
"Write method:\n " + 
writeMethod.toString()); 


System.out.println("====================") ; 


} 
System.out.println("Public methods:"); 
MethodDescriptor[] methods = 
bi.getMethodDescriptors(); 
for(int i = 0; i < methods.length; i++) 
System.out.printin( 
methods[i].getMethod().toString()); 
System. out.println("======================"); 
System.out.printin("Event support:"); 
EventSetDescriptor[] events = 
bi.getEventSetDescriptors(); 
for(int i = 0; i < events.length; i++) { 
System.out.printin("Listener type:\n " + 
events[i].getListenerType().getName()); 
Method[] lm = 
events[i].getListenerMethods(); 
for(int j = 0; j < lm.length; j++) 
System.out.printin( 
"Listener method:\n " + 
im[j].getName()); 
MethodDescriptor[] lmd = 
events[1i].getListenerMethodDescriptors(); 


for(int j = 0; j < 1Imd.length; j++) 


System.out .printlLn( 
"Method descriptor:\n " + 
imd[j].getMethod().toString()); 
Method addListener = 
events[i].getAddListenerMethod(); 
System.out.printin( 
"Add Listener Method:\n " + 
addListener.toString()); 
Method removeListener = 
events[i].getRemoveListenerMethod(); 
System.out.printin( 
"Remove Listener Method:\n " + 
removeListener.toString()); 


System.out.println("====================") ; 


// Dump the class of your choice: 


public static void main(String[] args) { 


if(args.length < 1) { 


System.err.printin("usage: \n" + 
"BeanDumper fully.qualified.class"); 


System.exit(0); 


Class c = null; 

try { 
c = Class.forName(args[0]); 

} catch(ClassNotFoundException ex) { 
System.err.printin( 

"Couldn't find " + args[0]); 

System.exit(0); 

} 

dump(c); 

} 
} ///:~ 





BeanDumper.dumpO 是 一 个 可 以 做 任何 工作 的 方法 。 首 先 它 试图 创建 一 
个 BeanInfo 对 象 ， 如 果 成 功 地 调用 BeanInfo 的 方法 ， 束 产生 关于 属性 、 
方法 和 事件 的 信息 。 在 Introspector.getBeanInfo0 中 ， 我 们 会 注意 到 有 一 
个 另外 的 自 变 量 。 由 它 来 通知 Introspector 访 问 继承 体系 的 地 点 。 在 这 种 
它 在 分 析 所 有 对 象 方法 前 停 下 ， 因 为 我 们 对 看 到 那些 并 不 感 兴 
Wo 


因为 属性 ，getPropertyDescriptorsO 返 回 一 组 的 属性 描述 符号 。 对 于 每 个 
描述 符号 我 们 可 以 调用 getPropertyType() 方 法 彻底 的 通过 属性 方法 发 现 
类 的 对 象 。 这 时 ， 我 们 可 以 用 getName0) 方 法 得 到 每 个 属性 的 假名 〈 从 
方法 名 中 提取 ) ，getname() 方 法 用 getReadMethod() 和 getWriteMethod() 
完成 读 和 写 的 操作 。 最 后 的 两 个 方法 返回 一 个 可 以 真正 地 用 来 调用 在 对 
象 上 调用 相应 的 方法 方法 对 象 〈 这 是 映 象 的 一 部 分 ) 。 对 于 公共 方法 
(包括 属性 方法 ) » getMethodDescriptors( ) 返回 一 组 方法 描述 字符 。 
每 一 个 我 们 都 可 以 得 到 相当 的 方法 对 象 并 可 以 显示 出 它们 的 名 字 。 


对 于 事件 而 言 ，getEventSetDescriptors() 返 回 一 组 事件 描述 字符 。 它 们 中 
的 每 一 个 都 可 以 被 查询 以 找 出 接收 器 的 类 ， 接 收 器 类 的 方法 以 及 增加 和 








删除 接收 器 的 方法 。BeanDumper 程 序 打 印 出 所 有 的 这 些 信息 。 
如 果 我 们 调用 BeanDumper 在 Frog 类 中 ， 束 像 这 样 : 
java BeanDumper frogbean.Frog 


它 的 输出 结果 如 下 已 删除 这 儿 不 需要 的 额外 细节 )〉: 


class name: Frog 


Property type: 
Color 
Property name: 
color 
Read method: 
public Color getColor() 
Write method: 


public void setColor(Color ) 


Property type: 
Spots 
Property name: 
spots 
Read method: 
public Spots getSpots() 
Write method: 


public void setSpots(Spots) 


Property type: 


boolean 


Property name: 


jumper 


Read method: 


public boolean isJumper() 


Write method: 


public void setJumper (boolean) 


Property type: 


int 


Property name: 


jumps 


Read method: 


public int getJumps() 


Write method: 


public void setJumps(int ) 


Public 
public 
public 


public 


methods: 
void setJumps(int) 
void croak() 


void removeActionListener (ActionListener ) 


public void addActionListener (ActionListener ) 
public int getJumps() 

public void setColor(Color) 

public void setSpots(Spots) 

public void setJumper (boolean) 

public boolean isJumper() 

public void addKeyListener(KeyListener ) 
public Color getColor() 

public void removeKeyListener(KeyListener ) 


public Spots getSpots() 


Event support: 
Listener type: 
KeyListener 
Listener method: 
keyTyped 
Listener method: 
keyPressed 
Listener method: 
keyReleased 
Method descriptor: 
public void keyTyped(KeyEvent ) 


Method descriptor: 


public void keyPressed(KeyEvent ) 
Method descriptor: 

public void keyReleased(KeyEvent ) 
Add Listener Method: 

public void addKeyListener(KeyListener ) 
Remove Listener Method: 


public void removeKeyListener(KeyListener ) 


Listener type: 
ActionListener 
Listener method: 
actionPerformed 
Method descriptor: 
public void actionPerformed(ActionEvent ) 
Add Listener Method: 
public void addActionListener (ActionListener ) 
Remove Listener Method: 


public void removeActionListener (ActionListener ) 


这 个 结果 揭示 出 了 Introspector 在 从 我 们 的 Bean 产 生 一 个 BeanInfo 对 象 时 
看 到 的 大 部 分 内 容 。 我 们 可 注意 到 属性 的 类 型 和 它们 的 名 字 是 相互 独立 
的 。 请 注意 小 写 的 属性 名 。 ( 当 属 性 名 开头 在 一 行 中 有 超过 不 止 的 大 写 
字母 ， 这 一 次 程序 就 不 会 被 执行 。) 并 且 请 记 住 我 们 在 这 里 所 见 到 的 方 
法 名 (例如 读 和 与 方法 ) 真正 地 从 一 个 可 以 被 用 来 在 对 象 中 调用 相关 方 


法 的 方法 对 象 中 产生 。 


通用 方法 列表 包含 了 不 相关 的 事件 或 者 属性 ， 例 如 croak()。 列 表 中 所 有 
的 方法 都 是 我 们 可 以 有 计划 的 为 Bean 调 用 ， 并 且 应 用 程序 构建 工具 可 以 
选择 列 出 所 有 的 方法 ， 当 我 们 调用 方法 时 ， 减 轻 我 们 的 任务 。 


最 后 ， 我 们 可 以 看 到 事件 在 接收 器 中 完全 地 分 析 研 究 它 的 方法 、 增 加 和 
减少 接收 器 的 方法 。 基 本 上 ， 一 旦 我 们 拥有 BeanInfo， 我 们 就 可 以 找 出 
对 Bean 来 说 任何 重要 的 事物 。 我 们 同样 可 以 为 Bean 调 用 方法 ， 即 使 我 们 
除了 对 象 外 没有 任何 其 它 的 信息 (此 外 ， 这 也 是 映 象 的 特点 ) 。 


13.18.3 一 个 更 复杂 的 Bean 


接 下 的 程序 例子 稍微 复杂 一 些 ， 尽 管 这 没有 什么 价值 。 这 个 程序 是 一 张 
不 论 鼠 标 何 时 移动 都 玮 绕 它 国 一 个 小 圆 的 IT inc HS See GS HD 
DME SCE te ol Kang”, FFA —SOIVER AE BG. HV. FY 
下 鼠标 键 时 ， 我 们 可 以 改变 的 属性 是 圆 的 大 小 ， 除 此 之 外 还 有 被 显示 文 
字 的 色彩 ， 大 小 ， 内 容 。BangBean 同 样 拥有 它 自 己 的 
addActionListener0 和 removeActionListener() 方 法 ， 因 此 我 们 可 以 附 上 自 
己 的 当 用 户 单 击 在 BangBean 上 时 会 被 激活 的 接收 器 。 这 样 ， 我 们 将 能 够 
确认 可 文 持 的 属性 和 事件 : 


//: BangBean. java 














// A graphical Bean 

package bangbean; 

import java.awt.*; 

import java.awt.event.*; 

import java.io.*; 

import java.util.*; 

public class BangBean extends Canvas 


implements Serializable { 


protected int xm, ym; 


protected int cSize 20; // Circle size 
protected String text = "Bang!"; 
protected int fontSize = 48; 
protected Color tColor = Color.red; 
protected ActionListener actionListener,; 
public BangBean() { 
addMouseListener(new ML()); 
addMouseMotionListener(new MML()); 
} 
public int getCircleSize() { return cSize; } 


public void setCircleSize(int newSize) { 


cSize = newSize; 


public String getBangText() { return text; } 
public void setBangText(String newText) { 


text = newText; 


public int getFontSize() { return fontSize; } 
public void setFontSize(int newSize) { 
fontSize = newSize; 


} 


public Color getTextColor() { return tColor; } 


public void setTextColor(Color newColor) { 
tColor = newColor; 

} 

public void paint(Graphics g) { 
g.setColor(Color.black); 
g.drawOval(xm - cSize/2, ym - cSize/2, 

cSize, cSize); 
} 


// This is a unicast listener, which is 


// the simplest form of listener management: 


public void addActionListener ( 


ActionListener 1) 
throws TooManyListenersException { 
if(actionListener != null) 
throw new TooManyListenersException(); 
actionListener = l; 
} 
public void removeActionListener ( 
ActionListener 1) { 
actionListener = null; 
} 
class ML extends MouseAdapter { 


public void mousePressed(MouseEvent e) { 


Graphics g = getGraphics(); 
g.setColor(tColor); 
g.setFont ( 
new Font( 
"TimesRoman", Font.BOLD, fontSize)); 
int width = 
g.getFontMetrics().stringwidth(text); 
g.drawString(text, 
(getSize().width - width) /2, 
getSize().height/2); 
g.dispose(); 
// Call the listener's method: 
if(actionListener != null) 
actionListener.actionPer formed ( 
new ActionEvent(BangBean.this, 


ActionEvent.ACTION_PERFORMED, null)); 


i 


class MML extends MouseMotionAdapter { 
public void mouseMoved(MouseEvent e) { 


xm = e.getXx(); 


ym e.getY(); 


repaint(); 


public Dimension getPreferredSize() { 
return new Dimension(200, 200); 
} 
// Testing the BangBean: 
public static void main(String[] args) { 
BangBean bb = new BangBean(); 
try { 
bb.addActionListener(new BBL()); 
} catch(TooManyListenersException e) {} 
Frame aFrame = new Frame("BangBean Test"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


aFrame.add(bb, BorderLayout ,CENTER ) ; 
aFrame.setSize(300, 300); 
aFrame.setVisible(true); 


} 


// During testing, send action information 


// to the console: 
static class BBL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


System.out.printin("BangBean action"); 


} 
} ///:~ 


最 重要 的 是 我 们 会 注意 到 BangBean 执 行 了 这 种 串联 化 的 接口 。 这 意味 着 
应 用 程序 构建 工具 可 以 在 程序 设计 者 调整 完 属性 值 后 利用 串联 为 
BangBean 贮 藏 所 有 的 信息 。 当 Bean 作 为 运行 的 应 用 程序 的 一 部 分 被 创 
建 时 ， 那 些 被 贮藏 的 属性 被 重新 恢复 ， 因 此 我 们 可 以 正确 地 得 到 我 们 的 


设计 。 


我 们 能 看 到 通常 同 Bean 一 起 运行 的 所 有 的 字段 都 是 专用 的 一 一 允许 只 能 
通过 方法 来 访问 ， 通 常 利用 “属性 ”结构 。 


当 我 们 注视 着 addActionListener() 的 签名 时 ， 我 们 会 注意 到 它 可 以 产生 出 
一 个 TooManyListenerException 〈 太 多 接收 器 异常 ) 。 这 个 异常 指明 它 

是 一 个 单一 的 类 型 的 ， 意 味 着 当 事 件 发 生 时 ， 它 只 能 通知 一 个 接收 器 。 

一 般 情 况 下 ， 我 们 会 使 用 具有 多 种 类 型 的 事件 ， 以 便 一 个 事件 通知 多 个 
的 接收 器 。 但 是 ， 那 样 会 陷入 直到 下 一 章 我 们 才能 准备 好 的 结局 中 ， 
此 这 些 内 容 会 被 重新 回顾 《下 一 个 标题 是 “Java Beans 的 重新 回顾 ”) 。 
单一 类 型 的 事件 回避 了 这 个 难题 。 


当 我 们 按 下 鼠标 键 时 ， 文 字 被 安 入 BangBean 中 间 ， 并 且 如 果 动 作 接收 器 
字段 存在 ， 它 的 actionPerformed() 方 法 就 被 调用 ， 创 建 一 个 新 的 
ActionEvent 对 象 在 处 理 过程 中 。 无 论 何 时 鼠标 移动 ， 它 的 新 座 标 将 被 捕 
所， 并 且 男 布 会 被 重 画 〈 像 我 们 所 看 到 的 抹 去 一 些 画 布 上 的 文字 ) 。 


main(0) 方 法 增加 了 人 允许 我 们 从 命令 行 中 测试 程序 的 功能 。 当 一 个 Bean 在 
一 个 开发 环境 中 ，main() 方 法 不 会 被 使 用 ， 但 拥有 它 是 绝对 有 益 的 ， 
为 它 提 供 了 快捷 的 测试 能 力 。 无 论 何 时 一 个 ActionEvent 发 生 ，main0) 方 




















法 都 将 创建 了 一 个 帧 并 安置 了 一 个 BangBean 在 它 里 面 ， 还 在 BangBean 
中 附 上 了 一 个 简单 的 动作 接收 器 以 打印 到 控制 台 。 当 然 ， 一 般 来 说 应 用 
程序 构建 工具 将 创建 大 多 数 的 Bean 的 代码 。 当 我 们 通过 BeanDumper 或 
者 安放 BangBean 到 一 个 可 激活 Bean 的 开发 环境 中 去 运行 BangBean 时 ， 
我 们 会 注意 到 会 有 很 多 额外 的 属性 和 动作 明显 超过 了 上 面 的 代码 。 那 是 
因为 BangBean 从 画布 中 继承 ， 并 且 男 布 就 是 一 个 Bean， 因 此 我 们 看 到 
它 的 属性 和 事件 同样 的 合适 。 


13.18.4 Bean 的 封装 


在 我 们 可 以 安放 一 个 Bean 到 一 个 可 激活 Bean 的 可 视 化 构建 工具 中 前 ， 它 
必须 被 放 入 到 标准 的 Bean 容 器 里 ， 也 就 是 包 仿 Bean 类 和 一 个 表示 “这 是 
一 个 Bean” 的 清单 文件 的 JAR (Java ARchive，Java 文 件 ) 文件 中 。 清 单 
文件 是 一 个 简单 的 紧 随 事件 结构 的 文本 文件 。 对 于 BangBean 而 言 ， 清 音 
文件 就 像 下 面 这 样 : 











Manifest-Version: 1.0 
Name: bangbean/BangBean.class 
Java-Bean: True 


其 中 ， 第 一 行 指出 清单 文件 结构 的 版 本 ， 这 是 SUN 公 司 在 很 久 以 前 公布 
的 版 本 。 第 二 行 〈 空 行 忽略 ) 对 文件 命名 为 BangBean.class。 第 三 行 表 
示 “ 这 个 文件 是 一 个 Bean”。 没 有 第 三 行 ， 程 序 构建 工具 不 会 将 类 作为 一 
个 Bean 来 认可 。 


唯一 难以 处 理 的 部 分 是 我 们 必须 肯定 “Name:” 字 段 中 的 路 径 是 正确 的 。 

如 果 我 们 回顾 BangBean.java， 我 们 会 看 到 它 在 package bangbean (AIA 
存放 类 路 径 的 子 目 录 称 为 "bangbean”) 中 ， 并 且 这 个 名 字 在 清单 文件 中 
必须 包括 封装 的 信息 。 另 外 ， 我 们 必须 安放 清单 文件 在 我 们 封装 路 径 的 
根 目录 上 ， 在 这 个 例子 中 意味 着 安放 文件 在 bangbean 子 目录 中 。 这 之 
后 ， 我 们 必须 从 同一 目录 中 调用 Jar 来 作为 清单 文件 ， 如 下 所 示 : 














jar cfm BangBean.jar BangBean.mf bangbean 


这 个 例子 假定 我 们 想 产 生 一 个 名 为 BangBean.jar 的 文件 并 且 我 们 将 清单 
放 到 一 个 称 为 BangBean.mf 文 件 中 。 





我 们 可 能 会 想 “ 当 我 编译 BangBean.java 时 ， 产 生 的 其 它 类 会 怎么 样 

We? ? 哦 ， 它 们 会 在 bangbean 子 目录 中 被 中 止 ， 并 且 我 们 会 注意 到 上 面 
jar 命 令 行 的 最 后 一 个 自 变 量 就 是 bangbean 子 目录 。 当 我 们 给 jar 子 目录 名 
时 ， 它 封装 整个 的 子 目 录 到 jar 文 件 中 《在 这 个 例子 中 ， 包 括 
BangBean.java 的 源 代 码 文件 一 一 对 于 我 们 自己 的 Bean 我 们 可 能 不 会 去 选 
择 包含 源 代码 文件 。) 另外 ， 如 果 我 们 改变 主意 ， 解 开打 包 的 JAR 文 
件 ， 我 们 会 发 现 我 们 清单 文件 并 不 在 里 面 ， 但 jar 创 建 了 它 自 己 的 清单 文 
件 ( 部 分 根据 我 们 的 文件 ) ， 称 为 MAINFEST.MF 并 且 安 放 它 到 META- 
INF 子 目录 中 (代表 “meta-information”) 。 如 果 我 们 打开 这 个 清单 文 
件 ， 我 们 同样 会 注意 到 jar 为 每 个 文件 加 入 数字 签名 信息 ， 其 结构 如 下 : 


Digest-Algorithms: SHA MD5 











SHA-Digest: pDpEAGINaeCx8aFtqPI4udSX/O0= 
MD5-Digest: O4NcS1hE3Smnzlp2hj6qeg== 


一 般 来 说 ， 我 们 不 必 担 心 这 些 ， 如 果 我 们 要 做 一 些 修 改 ， 可 以 修改 我 们 
的 原始 的 清单 文件 并 且 重 新 调用 jar 以 为 我 们 的 Bean 创 建 了 一 个 新 的 JAR 
文件 。 我 们 同样 也 可 以 简单 地 通过 增加 其 它 的 Bean 的 信息 到 我 们 清单 文 
件 来 增加 它们 到 JAR 文 件 中 。 


值得 注意 的 是 我 们 或 许 需要 安放 每 个 Bean 到 它 上 自己 的 子 目 录 中 ， 因 为 当 
我 们 创建 一 个 JAR 文 件 时 ， 分 配 JAR 应 用 目录 名 并 且 JAR 放 置 子 目录 中 
中 。 我 们 可 以 看 到 Frog 和 BangBean 都 在 它们 目 己 
的 子 目 录 中 。 


一 旦 我 们 将 我 们 的 Bean 正 确 地 放 入 一 个 JAR 文 件 中 ， 我 们 就 可 以 携带 它 
到 一 个 可 以 激活 Bean 的 编程 环境 中 使 用 。 使 用 这 种 方法 ， 我 们 可 以 从 一 
种 工具 到 男 一 种 工具 间 交 蔡 变 换 ， 但 SUN 公 司 为 Java Beans 提 供 了 免费 
高 效 的 测试 工具 在 它们 的 “Bean Development Kit，Bean 开 发 工 
H” (BDK) 称 为 “beanbox”。 “我们 可 以 从 www.javasoft.com 处 下 

载 。) 在 我 们 启动 beanbox 前 ， 放 置 我 们 的 Bean 到 beanbox 中 ， 复 制 JAR 
文件 到 BDK 的 “jars” 子 目录 中 。 


13.18.5 更 复杂 的 Bean 支 持 


我 们 可 以 看 到 创建 一 个 Bean 显 然 多 么 的 简单 。 在 程序 设计 中 我 们 几乎 不 














受到 任何 的 限制 。Java Bean 的 设计 提供 了 一 个 简单 的 输入 点 ， 这 样 可 以 
提高 到 更 复杂 的 层次 上 。 这 些 高 层次 的 问题 超出 了 这 本 书 所 要 讨论 的 范 
围 ， 但 它们 会 在 此 做 简要 的 介绍 。 我 们 可 以 在 http:Vjava.sun.comy/beans 

上 找到 更 多 的 详细 资料 。 


我 们 增加 更 加 复杂 的 程序 和 它 的 属性 到 一 个 位 置 。 上 面 的 例子 显示 一 个 
独特 的 属性 ， 当 然 它 也 可 能 代表 一 个 数组 的 属性 。 这 称 为 索引 属性 。 我 
们 简单 地 提供 一 个 相应 的 方法 〈 再 者 有 一 个 方法 名 的 命名 规则 ) 并且 
Introspector 认 可 索引 属性 ， 因 此 我 们 的 应 用 程序 构建 工具 相应 的 处 理 。 


属性 可 以 被 捆绑 ， 这 意味 着 它们 将 通过 PropertyChangeEvent 通 知 其 它 的 
对 象 。 其 它 的 对 象 可 以 随后 根据 对 Bean 的 改变 选择 修改 它们 自己 。 


属性 可 以 被 束缚 ， 这 意味 着 其 它 的 对 象 可 以 在 一 个 属性 的 改变 不 能 被 接 
受 时 ， 拒 绝 它 。 | 并 且 
它们 产生 一 个 ProptertyVetoException 去 阻止 修改 的 发 生 ， 并 恢复 为 原来 
的 值 。 


我 们 同样 能 够 改变 我 们 的 Bean 在 设计 时 的 被 描绘 成 的 方法 : 


(1) 我 们 可 以 为 我 们 特殊 的 Bean 提 供 一 个 定制 的 属性 表 。 普通 的 局 
性 表 将 被 所 有 的 Bean 所 使 用 ， 但 当 我 们 的 Bean 被 选择 时 ， A ere 
用 这 张 属性 表 。 


(2) 我 们 可 以 为 一 个 特殊 的 属性 创建 一 个 定制 的 编辑 器 ， 因 此 普通 的 属 
性 表 被 使 用 ， 但 当 我 们 指定 的 属性 被 调用 时 ， 编 辑 器 会 自动 地 航 调 用 。 


(3) 我 们 可 以 为 我 们 的 Bean 提 供 一 个 定制 的 BeanInfo 类 ， 产 生 的 信息 不 同 
于 由 Introspector 默 认 产 生 的 


(4) 它 同 样 可 能 在 所 有 的 FeatureDescriptors 中 改变 “expert” 的 开关 模式 ， 
以 辨别 基本 特征 和 更 复杂 的 特征 。 


13.18.6 Bean 更 多 的 知识 
另外 有 关 的 争议 是 Bean 不 能 被 编 址 。 无 论 何 时 我 们 创建 一 个 Bean， 都 希 


望 它 会 在 一 个 多 线程 的 环境 中 运行 。 这 意味 着 我 们 必须 理解 线程 的 出 
口 ， 我 们 将 在 下 一 章 中 介绍 。 我 们 会 发 现 有 一 段 称 为 “Java “Beans 的 回 














顾 * 的 市 会 注意 到 这 个 问题 和 它 的 解决 方 采 。 


13.19 Swing 入 门 〈 注 释 @) 


通过 这 一 章 的 学 习 ， 当 我 们 的 工作 方法 在 AWT 中 发 生 了 巨大 的 改变 后 

(如 果 可 以 回忆 起 很 久 以 前 ， 当 Java 第 一 次 面世 时 SUN 公 司 曾 声明 Java 
是 一 种 “稳定 ， 牢 固 ” 的 编程 语言 ) ， 可 能 一 直 有 Java 还 不 十 分 的 成 熟 的 
感觉 。 的 确 ， 现 在 Java 拥 有 一 个 不 错 的 事件 模型 以 及 一 个 优秀 的 组 件 复 
JavaBeans。 但 GUI 组 件 看 起 来 还 相当 的 原始 ， 笨 拙 以 及 相当 


O: 写作 本 节 时 ，Swing 库 显然 已 被 Sun“ 固 定 ”* 下 来 了 ， 所 以 只 要 你 下 载 
并 安 狼 了 Swing 库 ， 束 应 该 能 正确 地 编译 和 运行 这 里 的 代码 ， 不 会 出 现 
任何 问题 (应 该 能 编译 Sun 配 套 提供 的 演示 程序 ， 以 检测 安装 是 否 

确 ) 。 若 遇 到 任何 麻烦 ， 请 访问 http:/www.BruceEckel.com， 了 解 最 近 
的 更 新 情况 。 


而 这 就 是 Swing 将 要 占领 的 领域 。Swing 库 在 Java 1.1 之 后 面世 ， 因 此 我 
们 可 以 自然 而 然 地 假设 它 是 Java ”1.2 的 一 部 分 。 可 是 ， 它 是 设计 为 作为 
一 个 补充 在 Java 1.1 版 中 工作 的 。 这 样 ， 我 们 就 不 必 为 了 享用 好 的 UI 组 
件 库 而 等 待 我 们 的 平台 去 支持 Java 1.2 版 了 。 如 果 Swing 库 不 是 我 们 的 用 
户 的 Java ”1.1 版 所 支持 的 一 部 分 ， 并 且 产 生 一 些 意 外 ， 那 他 就 可 能 真正 
的 需要 去 下 载 Swing 库 了 。 


Swing 包含 所 有 我 们 缺乏 的 组 件 ， 在 整个 本 章 余 下 的 部 分 中 : 我 们 期 望 
领会 现代 化 的 UI， 来 目 按钮 的 任何 事件 包括 到 树 状 和 网 格 结构 中 的 图 
片 。 它 是 一 个 大 库 ， 但 在 茶 些 方面 它 为 任务 被 设计 得 相应 的 复杂 一 一 如 
打 任 何事 都 是 简单 的 ， 我 们 不 必 有 编写 更 多 的 代码 但 同样 设法 运行 我 们 的 
代码 逐渐 地 变 得 更 加 的 复杂 。 这 意味 独 一 个 容易 的 入 口 ， 如 采 我 们 需要 
它 我 们 得 到 它 的 强大 力量 。 


Swing 相 当 的 深奥 ， 这 一 节 不 会 去 试图 让 读者 理解 ， 但 会 介绍 它 的 能 

和 Swing 简单 地 使 我 们 着 手 使 用 库 。 请 注意 我 们 有 意识 的 使 用 这 一 切 变 
得 简单 。 如 果 我 们 需要 运行 更 多 的 ， 这 时 Swing 能 或 许 能 给 我 们 所 想 要 
He E 可 以 从 SUN 公 司 的 在 线 文档 中 获取 更 多 


13.19.1 Swing 有 哪些 优点 
































当 我 们 开始 使 用 Swing 库 时 ， 会 注意 到 它 在 技术 上 癌 前 迈 出 了 巨大 的 一 
步 。Swing 组 件 是 Bean， 因 此 他 们 可 以 支持 Bean 的 任何 开发 环境 中 使 
用 。Swing 提 供 了 一 个 完全 的 UI 组 件 集合 。 因 为 速度 的 关系 ， 所 有 的 组 
件 都 很 小 巧 的 (没有 “重量 级 ”组 件 被 使 用 ) ，Swing 为 了 轻便 在 Java 中 
整个 被 编写 。 


最 重要 的 是 我 们 会 希望 Swing 被 称 为 “ 正 交 使 用 ”;， 一旦 我 们 采用 了 这 种 
关于 库 的 普 训 的 办 法 我 们 残 可 以 在 任何 地 方 应 用 它们 。 这 主要 是 因为 

Bean 的 命名 规则 ， 大 多 数 的 时 候 在 我 编写 这 些 程序 例子 时 我 可 以 猜 到 方 
法 名 并 且 第 一 次 就 将 它 拼 写 正 确 而 无 需 碍 找 任何 事物 。 这 无 疑 是 优秀 库 
设计 的 品质 证 明 。 妇 外 ， 我 们 可 以 广泛 地 插入 组 件 到 其 它 的 组 件 中 并 且 
事件 会 正常 地 工作 。 


键盘 操作 是 自动 被 支持 的 一 我 们 可 以 使 用 Swing 应 用 程序 而 不 需要 鼠 
标 ， 但 我 们 不 得 不 做 一 些 额 外 的 编程 工作 〈 老 的 AWT 中 需要 一 些 可 怕 
的 代码 以 文 持 键盘 操作 ) 。 滚 动 被 宫 不 费力 地 文 持 一 我们 简单 地 将 我 
们 的 组 件 到 一 个 JScrollPane 中 ， 同 样 我们 再 增加 它 到 我 们 的 窗 体 中 即 
可 。 其 它 的 特征 ， 例 如 工具 提示 条 只 需要 一 行 单独 的 代码 就 可 执行 。 
Swing 同 样 支持 一 些 被 称 为 “可 插入 外 观 和 效果 ”的 事物 ， 这 就 是 说 UI 的 
外 观 可 以 在 不 同 的 平台 和 不 同 的 操作 系统 上 被 动态 地 改变 以 符合 用 户 的 
期 望 。 它 甚至 可 以 创造 我 们 自己 的 外 观 和 效果 。 

13.19.2 方便 的 转换 

如 果 我 们 长 期 艰苦 不 懈 地 利用 Java 1.1 版 构建 我 们 的 UI， 我 们 并 不 需要 
扔 掉 它 改变 到 Swing 阵营 中 来 。 幸 运 的 是 ， 库 被 设计 得 允许 容易 地 修改 
一 一 在 很 多 情况 下 我 们 可 以 简单 地 放 一 个 “J" 到 我 们 老 AWT 组 件 的 每 个 
类 名 前 面 即 可 。 下 面 这 个 例子 拥有 我 们 所 熟悉 的 特色 : 


//: JButtonDemo.java 


























// Looks like Java 1.1 but with J's added 
package c13.Swing; 


import java.awt.*; 


import java.awt.event.*; 

import java.applet.*; 

import javax.swing.*; 

public class JButtonDemo extends Applet { 
JButton 


b1 


new JButton("JButton 1"), 


b2 


new JButton("JButton 2"); 
JTextField t = new JTextField(20); 
public void init() { 
ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e){ 
String name = 
((JButton)e.getSource()).getText(); 


t.setText(name + " Pressed"); 


}; 
b1.addActionListener(al); 
add(b1); 
b2.addActionListener(al); 
add(b2); 
add(t); 

} 


public static void main(String args[]) { 


JButtonDemo applet = new JButtonDemo(); 
JFrame frame = new JFrame("TextAreaNew" ); 
frame.addwWindowListener(new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
System.exit(0); 
} 
}); 


frame.getContentPane().add( 
applet, BorderLayout.CENTER) ; 

frame.setSize(300, 100); 
applet.init(); 

applet.start(); 
frame.setVisible(true); 

} 
} ///:~ 


这 是 一 个 新 的 输入 语句 ， 但 此 外 任何 事物 除了 增加 了 一 些 “J* 外 ， 看 起 
都 像 这 Java 1.1 版 的 AWT。 同 样 ， 我 们 不 恰当 的 用 add0) 方 法 增加 到 
Swing JFrame 中 ， 除 此 之 外 我 们 必须 像 上 面 看 到 的 一 样 先 准备 一 
aoe pane”。 我 们 可 以 容易 地 得 到 Swing 一 个 简单 的 改变 所 带 来 的 
了 处 。 


人 我 们 不 得 不 调用 像 下 面 所 写 的 一 样 调用 这 个 程 
P: 








java c13.swing.JbuttonDemo 


在 这 一 节 里 出 现 的 所 有 的 程序 都 将 需要 一 个 相同 的 窗 体 来 运行 它们 。 





13.19.3 显示 框架 


尽管 程序 片 和 应 用 程序 都 可 以 变 得 很 重要 ， 但 如 条 在 任何 地 方 都 使 用 它 
们 就 会 变 得 混乱 和 军 无 用 处 。 这 一 节余 下 部 分 取代 它们 的 是 一 个 Swing 
程序 例子 的 显示 框架 : 


//: Show,java 











// Tool for displaying Swing demos 
package c13.swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.Swing.*; 
public class Show { 
public static void 
inFrame(JPanel jp, int width, int height) { 
String title = jp.getClass().toString(); 
// Remove the word "class": 
if(title.indexOf("class") != -1) 
title = title.substring(6); 
JFrame frame = new JFrame(title); 
frame.addWindowListener(new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 


System.exit(0); 


}); 


frame.getContentPane().add( 
jp, BorderLayout.CENTER); 
frame.setSize(width, height); 
frame.setVisible(true); 
} 
} ///:~ 


那些 想 显 示 它 们 自己 的 类 将 从 JPanel 处 继承 并 且 随 后 为 它们 自己 增加 一 
些 可 视 化 的 组 件 。 最 后 ， 它 们 创建 一 个 包含 下 面 这 一 行程 序 的 main(): 
Show.inFrame(new MyClass(), 500, 300); 

最 后 的 两 个 自 变 量 是 显示 的 宽度 和 高 度 。 


注意 JFrame 的 标题 是 用 RITTI 产 生 的 。 





13.19.4 工具 提示 


几乎 所 有 我 们 利用 来 创建 我 们 用 户 接 口 的 来 自 于 JComponent 的 类 都 包 合 
一 个 称 为 setToolTipText(string) 的 方法 。 因 此 ， 几 乎 任何 我 们 所 需要 表 
eee 自 JComponent 的 类 ) 都 可 以 安放 
于 窗 体 中 : 


jc.setToolTipText("My tip"); 


并 且 当 鼠标 停 在 JComponent 上 一 个 超过 预先 设置 的 一 个 时 间 ， 一 个 包含 
我 们 的 文字 的 小 框 束 会 从 鼠标 下 弹出 。 


13.19.5 边框 


JComponent 同 样 包括 一 个 称 为 setBorder() 的 方法 ， 该 方法 允许 我 们 安放 
一 些 各 种 各 样 有 趣 的 边框 到 一 些 可 见 的 组 件 上 。 下 面 的 程序 例子 利用 一 
个 创建 JPanel 并 安放 边框 到 每 个 例子 中 的 被 称 为 ShowBorder0O 的 方法 ， 示 
范 了 一 些 有 用 的 不 同 的 边框 。 同 样 ， 它 也 使 用 RTTI 来 找 我 们 使 用 的 边 








框 名 《剔除 所 有 的 路 径 信 息 ) ， 然 后 将 边框 名 放 到 面板 中 间 的 JLable 
里 : 
//: Borders.java 


// Different Swing borders 
package c13.swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.swing.*; 
import javax.swing.border.*; 
public class Borders extends JPanel { 
static JPanel showBorder(Border b) { 
JPanel jp = new JPanel(); 
jp.setLayout(new BorderLayout()); 
String nm = b.getClass().toString(); 
nm = nm.substring(nm.lastIndexOf('.') + 1); 
jp.add(new JLabel(nm, JLabel.CENTER), 
BorderLayout.CENTER); 
jp.setBorder(b); 
return jp; 
} 
public Borders() { 


setLayout(new GridLayout(2,4)); 


add(showBorder(new TitledBorder("Title"))); 
add(showBorder(new EtchedBorder())); 
add(showBorder (new LineBorder(Color.blue))); 
add(showBorder ( 
new MatteBorder(5,5,30,30,Color.green))); 
add(showBorder ( 
new BevelBorder(BevelBorder .RAISED) ) ); 
add(showBorder ( 
new SoftBevelBorder (BevelBorder .LOWERED) ) ); 
add(showBorder (new CompoundBorder ( 
new EtchedBorder(), 
new LineBorder(Color.red)))); 
} 
public static void main(String args[]) { 
Show.inFrame(new Borders(), 500, 300); 
} 
} ///:~ 


这 一 节 中 大 多 数 程序 例子 都 使 用 TitledBorder， 但 我 们 可 以 注意 到 其 余 的 
边框 也 同样 易于 使 用 。 能 创建 我 们 自己 的 边框 并 安放 它们 到 按钮 、 标 签 
等 等 内 一 一 任何 来 自 JComponent 的 东西 。 


13.19.6 按钮 


Swing 增 加 了 一 些 不 同类 型 的 按钮 ， 并 且 它 同样 可 以 修改 选择 组 件 的 结 
构 : 所 有 的 按钮 、 复 选 杠 、 单 选 钮 ， 其 至 从 AbstractButton 处 继承 的 菜单 











项 〈 这 是 因为 荣 单 项 一 般 被 包含 在 其 中 ， 它 可 能 会 被 改进 命名 
为 “AbstractChooser” 或 者 相同 的 什么 名 字 ) 。 我 们 会 注意 使 用 菜单 项 的 
简便 ， 下 面 的 例子 展示 了 不 同类 型 的 可 用 的 按钮 : 


//: Buttons.java 








// Narious Swing buttons 
package c13.Swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.Swing.*; 
import javax.swing.plaf.basic.*; 
import javax.Swing.border.*; 
public class Buttons extends JPanel { 
JButton jb = new JButton("JButton"); 
BasicArrowButton 
up = new BasicArrowButton( 
BasicArrowButton.NORTH), 
down = new BasicArrowButton( 
BasicArrowButton.SOUTH), 
right = new BasicArrowButton( 
BasicArrowButton.EAST), 
left = new BasicArrowButton( 
BasicArrowButton.WEST); 


public Buttons() { 


add(jb); 
add(new JToggleButton("JToggleButton") ); 
add(new JCheckBox("JCheckBox") ); 
add(new JRadioButton("JRadioButton") ); 
JPanel jp = new JPanel(); 
jp.setBorder(new TitledBorder("Directions")); 
jp.add(up); 
jp.add(down); 
jp.add(left); 
jp.add(right); 
add(jp); 

} 

public static void main(String args[]) { 
Show.inFrame(new Buttons(), 300, 200); 

} 

} ///:~ 





JButton 看 起 来 像 AWT 按 钮 ， 但 它 没 有 更 多 可 运行 的 功能 ( 像 我 们 后 面 
将 看 到 的 如 加 入 图 像 等 ) 。 在 com.sun.java.swing.basic 里 ， 有 一 个 更 合 
适 的 BasicArrowButton 按 钮 ， 但 怎样 测试 它 呢 ?有 两 种 类 型 的 “指针 ” 恰 
好 请 求 箭头 按钮 使 用 : Spinner 修 改 一 个 中 断 值 ， 并 且 StringSpinner 通 过 
一 个 字符 串 数组 来 移动 ( 当 它 到 达 数 组 底部 时 ， 其 至 会 自动 地 封装 ) 。 
ActionListeners 附 着 在 币 头 按钮 上 展示 它 使 用 的 这 些 相 关 指 针 : 因为 它 
们 是 Bean， 我 们 将 期 竺 利用 方法 名 ， 正 好 捕捉 并 设置 它们 的 值 。 


当 我 们 运行 这 个 程序 例子 时 ， 我 们 会 发 现 触 及 按钮 保持 它 最 新 状态 ， 开 
或 时 关 。 但 复 选 框 和 单 选 钮 每 一 个 动作 都 相同 ， 选 中 或 没 选中 它们 从 





JToggleButton 处 继承 ) 。 
13.19.7 按钮 组 


如 有 果 我 们 想 单 选 钮 保持 “ 异 或 ”状态 ， 我 们 必须 增加 它们 到 一 个 按钮 组 
中 ， 这 几乎 同 老 AWT 中 的 方法 相同 但 更 加 的 灵活 。 在 下 面 将 要 证 明 的 
程序 例子 是 ， 一 些 AbstruactButton 能 被 增加 到 一 个 ButtonGroup 中 。 


为 避免 重复 一 些 代 码 ， 这 个 程序 利用 映射 来 生 不 同类 型 的 按钮 组 。 这 会 
在 makeBPanel 中 看 到 ，makeBPanel 创 建 了 一 个 按钮 组 和 一 个 JPanel， 并 
且 为 数组 中 的 每 个 String 就 是 makeBPanel 的 第 二 个 自 变 量 增加 一 个 类 对 
象 ， 由 它 的 第 一 个 目 变 量 进行 声明 : 


//: ButtonGroups.java 





// Uses reflection to create groups of different 
// types of AbstractButton. 
package c13.swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.swing.*; 
import javax.swing.border.*; 
import java.lang.reflect.*; 
public class ButtonGroups extends JPanel { 
static String[] ids = { 
"June", "Ward", "Beaver", 
"Wally", "Eddie", "Lumpy", 
}; 


static JPanel 


makeBPanel(Class bClass, String[] ids) { 
ButtonGroup bg = new ButtonGroup(); 
JPanel jp = new JPanel(); 
String title = bClass.getName()j; 
title = title.substring( 
title.lastIndexOf('.') + 1); 
jp.setBorder(new TitledBorder(title)); 
for(int i = 0; i < ids.length; i++) { 
AbstractButton ab = new JButton("failed"); 
try { 
// Get the dynamic constructor method 
// that takes a String argument: 
Constructor ctor = bClass.getConstructor ( 
new Class[] { String.class }); 
// Create a new object: 
ab = (AbstractButton)ctor.newInstance( 
new Object[]{ids[i]}); 
} catch(Exception ex) { 
System.out.println("can't create " + 
bClass); 
} 
bg.add(ab); 


jp.add(ab); 


} 
return jp; 

} 

public ButtonGroups() { 
add(makeBPanel(JButton.class, ids)); 
add(makeBPanel(JToggleButton.class, ids)); 
add(makeBPanel(JCheckBox.class, ids)); 
add(makeBPanel(JRadioButton.class, ids)); 

} 

public static void main(String args[]) { 
Show.inFrame(new ButtonGroups(), 500, 300); 

} 

} ///:~ 


边框 标题 由 类 名 剔除 了 所 有 的 路 径 信息 而 来 。AbstractButton 初 始 化 为 一 

个 JButton，JButtonr 的 标签 有 发生“ 失效 >”， 因 此 如 果 我 们 忽略 这 个 异 背 信 

> 我 们 会 在 屏幕 上 一 直 看 到 这 个 问题 。getConstructor0) 方 法 产生 了 一 
个 通过 getConstructor() 方 法 安放 目 变 量 数组 类 型 到 类 ee 的 构建 器 对 

象 然后 所 有 我 们 要 做 的 就 是 调用 newlInstance()， 通 过 它 一 个 数组 对 象 

包含 我 们 当前 的 自 变量 一 一 在 这 种 例子 中 ， 就 是 ids 数 组 中 的 字符 串 。 


这 样 增加 了 一 些 更 复杂 的 内 容 到 这 个 简单 的 程序 中 。 为 了 使 “ 异 或 ”行为 
拥有 按钮 ， 我 们 创建 一 个 按钮 组 并 增加 每 个 按钮 到 我 们 所 需 的 组 中 。 当 
我 们 运行 这 个 程序 时 ， 我 们 会 注意 到 所 有 的 按钮 除了 JButton 都 会 向 我 们 
展示 “ 异 或 ”行为 。 

13.19.8 图 标 


我 们 可 在 一 个 JLable 或 从 AbstractButton 处 继承 的 任何 事物 中 使 用 一 个 图 




















标 〈 包 括 JButton，JCheckbox，JradioButton 及 不 同类 型 的 JMenuItem)。 
利用 JLables 的 网 标 十 分 的 简单 容易 《我 们 会 在 随后 的 一 个 程序 例子 中 看 
到 ) 。 下 面 的 程序 例子 探索 了 我 们 可 以 利用 按钮 的 图 标 和 它们 的 衍生 物 
的 其 它 所 有 方法 。 


我 们 可 以 使 用 任何 我 们 需要 的 GIF 文 件 ， 但 在 这 个 例子 中 使 用 的 这 个 
GIF 文件 是 这 本 书 编 码 发 行 的 一 部 分 ， 可 以 在 www.BruceEckel.com 处 下 
载 来 使 用 。 为 了 打开 一 个 文件 和 随 之 市 来 的 图 像 ， 简 单 地 创建 一 个 图 标 
并 分 配 它 文件 名 。 从 那 时 起 ， 我 们 可 以 在 程序 中 使 用 这 个 产生 的 图 标 。 


//: Faces.java 


// Icon behavior in JButtons 
package c13.swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.swing.*; 
public class Faces extends JPanel { 
static Icon[] faces = { 
new ImageIcon("faceO.gif"), 
new ImageIcon("face1.gif"), 
new ImageIcon("face2.gif"), 
new ImageIcon("face3.gif"), 
new ImageIcon("face4.gif"), 
3; 
JButton 


jb = new JButton("JButton", faces[3]), 


jb2 = new JButton("Disable"); 
boolean mad = false; 
public Faces() { 
jb.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e){ 
if(mad) { 
jb.setIcon(faces[3]); 
mad = false; 
} else { 
jb.setIcon(faces[0]); 
mad = true; 
} 
jb.setVerticalAlignment(JButton.TOP) ; 


jb.setHorizontalAlignment(JButton.LEFT); 


jb.setRolloverEnabled(true) ; 
jb.setRolloverIcon(faces[1]); 
jb.setPressediIcon(faces[2]); 
jb.setDisablediIcon(faces[4]); 
jb.setToolTipText("Yow!"); 
add(jb); 


jb2.addActionListener(new ActionListener() { 


public void actionPerformed(ActionEvent e){ 
if(jb.isEnabled()) { 
jb.setEnabled(false); 
jb2.setText("Enable") ; 
} else { 
jb.setEnabled(true); 


jb2.setText("Disable"); 


}); 
add(jb2); 
} 
public static void main(String args[]) { 
Show.inFrame(new Faces(), 300, 200); 
} 
} ///:~ 


一 个 图 标 可 以 在 许多 的 构建 器 中 使 用 ， 但 我 们 可 以 使 用 setIcon() 方 法 增 
加 或 更 换 图 标 。 这 个 例子 同样 展示 了 当 事 件 发 生 在 JButton (或 者 一 些 
AbstractButton) 上 时 ， 为 什么 它 可 以 设置 各 种 各 样 的 显示 图 标 : 4 
JButton 被 按 下 时 ， 当 它 被 失效 时 ， 或 者 “ 滚 过 ”时 〈 上 鼠标 从 它 上 面 移动 过 
但 并 不 击 它 ) 。 我 们 会 注意 到 那 给 了 按钮 一 种 动画 的 感觉 。 


注意 工具 提示 条 也 同样 增加 到 按钮 中 。 


13.19.9 菜单 











菜单 在 Swing 中 做 了 重要 的 改进 并 且 更 加 的 灵活 一 一 例如 ， 我 们 可 以 在 
几乎 程序 中 任何 地 方 使 用 他 们 ， 包 括 在 面板 和 程序 片 中 。 语 法 同 它们 在 
老 的 AWT 中 是 一 样 的 ， 并 且 这 样 使 出 现在 老 AWT 的 在 新 的 Swing 也 出 现 
T: 我 们 必须 为 我 们 的 沫 单 艰难 地 编写 代码 ， 并 且 有 一 些 不 再 作为 资源 
文 持 菜 单 〈 其 它 事件 中 的 一 些 将 使 它们 更 易 转 换 成 其 它 的 编程 语言 ) 。 
另外 ， 某 单 代 码 相当 的 元 长 ， 有 时 还 有 一 些 混乱 。 下 面 的 方法 是 放置 所 
有 的 关于 每 个 菜单 的 信息 到 对 象 的 三 维 数组 里 (这 种 方法 可 以 放置 我 们 
想 处 理 的 任何 事物 到 数组 里 ) ， 这 种 方法 在 解决 这 个 问题 方面 领先 了 一 
步 。 这 个 二 维 数组 伞 妆 单 所 创建 ， 因 此 它 首 先 表 示 出 琳 单 名 ， 并 在 剩余 
的 列 中 表示 荣 单 项 和 它们 的 特性 。 我 们 会 注意 到 数组 列 不 必 保 持 一 至 

只 要 我 们 的 代码 知道 将 发 生 的 一 切 事 件 ， 每 一 列 都 可 以 完全 不 同 。 


//: Menus.java 











// A menu-building system; also demonstrates 
// icons in labels and menu items. 
package ci13.swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.Swing.*; 
public class Menus extends JPanel { 
static final Boolean 
bT = new Boolean(true), 
bF = new Boolean(false); 
// Dummy class to create type identifiers: 
static class MType { MType(int i) {} }; 
static final MType 


mi = new MType(1), // Normal menu item 


cb 


new MType(2), // Checkbox menu item 


rb 


new MType(3); // Radio button menu item 
JTextField t = new JTextField(10); 
JLabel 1 = new JLabel("Icon Selected", 
Faces.faces[0], JLabel.CENTER); 
ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
t.setText( 


((JMenuItem)e.getSource()).getText()); 


}; 
ActionListener a2 = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
JMenuItem mi = (JMenuItem)e.getSource(); 
1.setText(mi.getText()); 


1.setIcon(mi.getIcon()); 


}; 
// Store menu data as "resources": 
public Object[][] fileMenu = { 

// Menu name and accelerator: 


{ "File", new Character('F') }, 


// Name type accel listener enabled 


Fa 


a “`` M` 一 一 


"New", mi, new Character('N'), ai, bT }, 
"Open", mi, new Character('0'), al, bT }, 
"Save", mi, new Character('S'), al, bF }, 
"Save As", mi, new Character('A'), a1, bF}, 
null }, // Separator 


"Exit", mi, new Character('x'), atl, bT }, 


public Object[][] editMenu = { 


I; 


// Menu name: 


{ 


"Edit", new Character('E') }, 


// Name type accel listener enabled 


{ 


{ 
{ 
{ 
af 


"Cut", mi, new Character('t'), ai, bT }, 
"Copy", mi, new Character('C'), a1, bT }, 
"Paste", mi, new Character('P'), ai, bT }, 
null }, // Separator 


"Select All", mi,new Character('1l'),a1,bT}, 


public Object[][] helpMenu = { 


// Menu name: 


i 


"Help", new Character('H') }, 


// Name type accel listener enabled 


i 
i 


"Index", mi, new Character('I'), a1, bT }, 


"Using help", mi,new Character('U'),a1,bT}, 


{ null }, // Separator 
{ "About", mi, new Character('t'), ai, bT }, 
}; 
public Object[][] optionMenu = { 
// Menu name: 
{ "Options", new Character('0O') }, 
// Name type accel listener enabled 
{ "Option 1", cb, new Character('1'), a1,bT}, 
{ "Option 2", cb, new Character('2'), ai,bT}, 
}; 
public Object[][] faceMenu = { 
// Menu name: 
{ "Faces", new Character('a') }, 
// Optinal last element is icon 
{ "Face 0", rb, new Character('O'), a2, bT, 
Faces.faces[0] }, 
{ "Face 1", rb, new Character('1'), a2, bT, 
Faces.faces[1] }, 
{ "Face 2", rb, new Character('2'), a2, bT, 
Faces.faces[2] }, 
{ "Face 3", rb, new Character('3'), a2, bT, 
Faces.faces[3] }, 


{ "Face 4", rb, new Character('4'), a2, bT, 


Faces.faces[4] }, 
}; 
public Object[] menuBar = { 
fileMenu, editMenu, faceMenu, 
optionMenu, helpMenu, 
}; 
static public JMenuBar 
createMenuBar(Object[] menuBarData) { 
JMenuBar menuBar = new JMenuBar(); 
for(int i = 0; i < menuBarData.length; i++) 
menuBar . add( 
createMenu((Object[][])menuBarData[i])); 
return menuBar; 
} 
static ButtonGroup bgroup; 
static public JMenu 
createMenu(Object[][] menuData) { 
JMenu menu = new JMenu(); 
menu.setText((String)menuData[0][0]); 
menu. setMnemonic( 
((Character )menuData[0][1]).charValue()); 
// Create redundantly, in case there are 


// any radio buttons: 


bgroup = new ButtonGroup(); 
for(int i = 1; i < menuData.length; I++) { 
if(menuData[i][0] == null) 
menu.add(new JSeparator()); 
else 
menu.add(createMenuItem(menuData[i])); 


} 


return menu; 
} 
static public JMenuItem 
createMenulItem(Object[] data) { 
JMenuItem m = null; 
MType type = (MType)data[1]; 
if(type == mi) 
m = new JMenuItem(); 
else if(type == cb) 
m = new JCheckBoxMenultem(); 
else if(type == rb) { 
m = new JRadioButtonMenulItem(); 
bgroup.add(m); 
} 
m.setText((String)data[0]); 


m.setMnemonic( 


((Character )data[2]).charValue()); 
m.addActionListener ( 
(ActionListener )data[3]); 
m.setEnabled( 
((Boolean)data[4]).booleanValue()); 
if(data.length == 6) 
m.setIcon((Icon)data[5]); 
return m; 
} 
Menus() { 
setLayout(new BorderLayout()); 
add(createMenuBar (menuBar ), 
BorderLayout.NORTH) ; 
JPanel p = new JPanel(); 
p.setLayout(new BorderLayout()); 
p.add(t, BorderLayout.NORTH) ; 
p.add(1, BorderLayout.CENTER) ; 
add(p, BorderLayout.CENTER) ; 
} 
public static void main(String args[]) { 


Show.inFrame(new Menus(), 300, 200); 


de 


这 个 程序 的 目的 是 允许 程序 设计 者 简单 地 创建 表格 来 描述 每 个 沫 单 ， 而 
不 是 输入 代码 行 来 建立 沫 单 。 每 个 沫 单 都 产生 一 个 染 单 ， 表 格 中 的 第 一 
列 包含 沫 单 名 和 键盘 快捷 键 。 其 余 的 列 包含 每 个 亲 单 项 的 数据 ， 字符 串 
存在 在 沫 单项 中 的 位 置 ， 沫 单 的 类 型 ， 它 的 快捷 键 ， 当 沫 单项 被 选中 时 
被 激活 的 动作 接收 器 及 沫 单 是 否 被 激活 等 信息 。 如 果 列 开始 处 是 空 的 ， 
它 将 被 作为 一 个 分 隔 符 来 处 理 。 


为 了 预防 浪费 和 宛 长 的 多 个 Boolean 创 建 的 对 象 和 类 型 标志 ， 以 下 的 这 
些 在 类 开始 时 就 作为 static final 被 创建 : bT ADF FIA Booleans Fil MPs 
MTypelt) As [a] XY Be Fini PE SE ALI mi) ， 复 选 框 沫 单项 (cb) , Ail 
单 选 钮 菜单 项 (rb) 。 请 记 住 一 组 Object 可 以 拥有 单一 的 Object 句柄 ， 并 
且 不 再 是 原来 的 值 。 


这 个 程序 例子 同样 展示 了 JLables 和 JMenultems (和 它们 的 衍生 事物 ) 如 
何 处 理 图 标的 。 一 个 图 标 经 由 它 的 构建 右 置 放 进 JLable 中 并 当 对 应 的 羔 
单项 被 选中 时 被 改变 。 


沫 单条 数组 控制 处 理 所 有 在 文件 染 单 清单 中 列 出 的 ， 我 们 想 显 示 在 沈 单 
条 上 的 文件 菜单 。 我 们 通过 这 个 数组 去 使 用 createMenuBar()， 将 数组 分 
类 成 单独 的 衣 单 数据 数组 ， 青 通过 每 个 单独 的 数组 去 创建 菜单 。 这 种 方 
法 依次 使 用 染 单 数据 的 每 一 行 并 以 该 数据 创建 JMenu， 然 后 为 沫 单数 据 
中 剩 下 的 每 一 行 调 用 createMenuItem() 方 法 。 最 后 ，createMenuItem() 方 
法 分 析 沫 单数 据 的 每 一 行 并 且 判 断 沈 单 类 型 和 它 的 属性 ， 再 适当 地 创建 
KT. AT, REESE ESS FO BIE PE, NER 
createMenuBar(menuBan 的 表格 中 创建 菜单 ， 而 所 有 的 事物 都 是 采用 递 
VAATED ERA « 


这 个 程序 不 能 建立 串联 的 沫 单 ， 但 我 们 拥有 足够 的 知识 ， 如 果 我 们 需要 
的 话 ， 随 时 都 能 增加 多 级 菜单 进去 。 


13.19.10 弹出 式 菜单 


JPopupMenu 的 执行 看 起 来 有 一 些 别扭 : 我们 必须 调用 enableEvents() 方 
法 并 选择 鼠标 事件 代 蔡 利用 事件 接收 费 。 它 可 能 增加 一 个 鼠标 接收 器 但 
MouseEvent 从 isPopupTrigger() 处 不 会 返回 真 值 它 不 知道 将 激活 一 个 
弹出 菜单 。 男 外 ， 当 我 们 尝试 接收 器 方法 时 ， 它 的 行为 令 人 不 可 思议 ， 

















这 或 许 是 鼠标 单 击 活动 引起 的 。 在 下 面 的 程序 例子 里 一 些 事 件 产生 了 这 
种 弹出 行为 : 


//: Popup.java 


// Creating popup menus with Swing 
package c13.swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.swing.*; 
public class Popup extends JPanel { 
JPopupMenu popup = new JPopupMenu(); 
JTextField t = new JTextField(10); 
public Popup() { 
add(t); 
ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e){ 
t.setText( 


((JMenuItem)e.getSource()).getText()); 


hi 
JMenuItem m = new JMenuItem("Hither"); 
m.addActionListener(al); 


popup.add(m); 


m = new JMenuItem("Yon"); 
m.addActionListener(al); 
popup.add(m); 

m = new JMenuItem("Afar"); 
m.addActionListener (al); 
popup.add(m); 
popup.addSeparator(); 

m = new JMenuItem("Stay Here"); 
m.addActionListener (al); 
popup.add(m); 

PopupListener pl = new PopupListener(); 
addMouseListener(pl); 


t.addMouseListener(pl1); 


class PopupListener extends MouseAdapter { 


public void mousePressed(MouseEvent e) { 
maybeShowPopup(e) ; 

} 

public void mouseReleased(MouseEvent e) { 
maybeShowPopup(e); 

} 

private void maybeShowPopup(MouseEvent e) 


if(e.isPopupTrigger()) { 


popup. show( 


e.getComponent(), e.getX(), e.getY()); 


t 
public static void main(String args[]) { 
Show. inFrame(new Popup(), 200,150); 
} 
} ///:~ 


相同 的 ActionListener 被 加 入 每 个 JMenuItem 中 ， 使 其 能 从 菜单 标签 中 取 
出 文字 ， 并 将 文字 插入 JTextField。 


13.19.11 列表 框 和 组 合 框 


列表 框 和 组 合 框 在 Swing 中 工作 就 像 它 们 在 老 的 AWT 中 工作 一 样 ， 但 如 
果 我 们 需要 它 ， 它 们 同样 被 增加 功能 。 另 外 ， 它 也 更 加 的 方便 易 用 。 例 
如 ，JList 中 有 一 个 显示 String 数 组 的 构建 器 (奇怪 的 是 同样 的 功能 在 
JComboBox 中 无 效 ! ) 。 下 面 的 例子 显示 了 它们 基本 的 用 法 。 


//: ListCombo.java 











// List boxes & Combo boxes 
package c13.swing; 

import java.awt.*; 

import java.awt.event.*; 
import javax.swing.*; 


public class ListCombo extends JPanel { 


public ListCombo() { 
setLayout(new GridLayout(2,1)); 
JList list = new JList(ButtonGroups.ids); 
add(new JScrollPane(list)); 
JComboBox combo = new JComboBox(); 
for(int i = 0; i < 100; i++) 

combo.addItem(Integer.toString(i)); 

add(combo) ; 

} 

public static void main(String args[]) { 
Show.inFrame(new ListCombo(),200, 200); 

} 

} ///:~ 


最 开始 的 时 候 ， 似 乎 有 点 儿 古 怪 的 一 种 情况 是 JLists 大 然 不 能 自动 提供 
深 动 特性 一 一 即使 那 也 许 正 是 我 们 一 直 所 期 望 的 。 增 加 对 滚动 的 支持 变 
得 十 分 容易 ， 束 像 上 面 示范 的 一 样 简单 地 将 JList 封 装 到 JScrollPane 
即 可 ， 所 有 的 细节 都 自动 地 为 我 们 照料 到 了 。 


13.19.12 滑 杆 和 进度 指示 条 


滑 杆 用 户 能 用 一 个 滑 块 的 来 回 移动 来 输入 数据 ， 在 很 多 情况 下 吕 得 很 下 
观 〈 如 声音 控制 ) 。 进 程 条 从 *“ 空 ?到 * 满 ”显示 相关 数据 的 状态 ， 因 此 用 
户 得 到 了 一 个 状态 的 透视 。 我 最 喜爱 的 有 头 这 的 程序 例子 简单 地 将 请 动 
所 以 当 我 们 移动 请 动 块 时 ， 进 程 条 也 相应 的 改 


//: Progress.java 




















// Using progress bars and sliders 


package c13.swing; 


import 
import 
import 
import 
import 


public 


java.awt.*; 
java.awt.event.*,; 
javax.swing.*; 
javax.swing.event.*; 
javax.swing.border.*; 


class Progress extends JPanel { 


JProgressBar pb = new JProgressBar()j; 


JSlider sb = 


new JSlider(JSlider.HORIZONTAL, 0, 100, 60); 


public Progress() { 


setLayout(new GridLayout(2,1)); 


add(pb); 


sb 


sb. 


sb 


sb. 


sb. 


pb 


.setValue(0); 

setPaintTicks(true); 
.setMajorTickSpacing(20); 
setMinorTickSpacing(5); 

setBorder(new TitledBorder("Slide Me")); 


.setModel(sb.getModel()); // Share model 


add(sb); 


} 


public static void main(String args[]) { 


Show.inFrame(new Progress(),200,150); 


} ///:~ 


JProgressBar 十 分 简单 ， 但 JSlider 却 有 许多 选项 ， 例 如 方法 、 大 或 小 的 记 
号 标签 。 注 意 增加 一 个 带 标 题 的 边框 是 多 么 的 容易 。 


13.19.13 树 
使 用 一 个 JTree 可 以 简单 地 像 下 面 这 样 表示 : 
add(new JTree( 


new Object[] {"this", "that", "other"})); 


这 个 程序 显示 了 一 个 原始 的 树 状 物 。 树 状 物 的 API 是 非常 巨大 的 ， 可 是 
一 一 当然 古 在 Swing 中 的 巨大 。 它 表明 我 们 可 以 做 有 关 树 状 物 的 任何 
事 ， 但 更 复杂 的 任务 可 能 需要 不 少 的 研究 和 试验 。 笠 运 的 是 ， 在 库 中 提 
供 了 一 个 妥协 : “默认 的 ” 树 状 物 组 件 ， 通 营 那 是 我 们 所 需要 的 。 因 此 大 
多 数 的 时 间 我 们 可 以 利用 这 些 组 件 ， 并 且 只 在 特殊 的 情况 下 我 们 需要 更 
深入 的 研究 和 理解 。 


下 面 的 例子 使 用 了 “默认 ”的 树 状 物 组 件 在 一 个 程序 片 中 显示 一 个 树 状 
物 。 当 我 们 按 下 按钮 时 ， 一 个 新 的 子 树 就 被 增加 a 到 当前 选中 的 结 皮 下 
《如果 没有 结 扣 被 选中 ， 束 用 根 结 市 〉: 


//: Trees.java 


// Simple Swing tree example. Trees can be made 
// vastly more complex than this. 

package c13.swing; 

import java.awt.*; 


import java.awt.event.*; 


import javax.swing.*; 
import javax.swing.tree.*; 
// Takes an array of Strings and makes the first 
// element a node and the rest leaves: 
class Branch { 
DefaultMutableTreeNode r; 
public Branch(String[] data) { 
r = new DefaultMutableTreeNode(data[0]); 
for(int i = 1; i < data.length; i++) 
r.add(new DefaultMutableTreeNode(data[i])); 
} 
public DefaultMutableTreeNode node() { 


return r; 


} 


public class Trees extends JPanel { 
String[][] data = { 

{ "Colors", "Red", "Blue", "Green" }, 
"Flavors", "Tart", "Sweet", "Bland" }, 
"Length", "Short", "Medium", "Long" }, 
"Volume", "High", "Medium", "Low" }, 


"Temperature", "High", "Medium", "Low" }, 


rr 一 en 一 


"Intensity", "High", "Medium", "Low" }, 


}; 


static int i = 0; 


DefaultMutableTreeNode root, child, chosen; 


JTree tree; 


DefaultT 


public T 


reeModel model; 


rees() { 


setLayout(new BorderLayout()); 


root = 


tree = 


// Add 


new DefaultMutableTreeNode("root"); 
new JTree(root); 


it and make it take care of scrolling: 


add(new JScrollPane(tree), 


Bord 
// Cap 
model 


JButto 


erLayout , CENTER ) ， 
ture the tree's model: 
=(DefaultTreeModel)tree.getModel(); 


n test = new JButton("Press me"); 


test.addActionListener(new ActionListener() { 


public void actionPerformed(ActionEvent e){ 


if(i < data.length) { 


child = new Branch(data[i++]).node(); 

// What's the last one you clicked? 

chosen = (DefaultMutableTreeNode) 
tree.getLastSelectedPathComponent(); 


if(chosen == null) chosen = root; 


// The model will create the 
// appropriate event. In response, the 
// tree will update itself: 
model.insertNodeInto(child, chosen, 0); 
// This puts the new node on the 
// currently chosen node. 
} 
} 
}); 
// Change the button's colors: 
test.setBackground(Color.blue); 
test.setForeground(Color.white); 
JPanel p = new JPanel(); 
p.add(test); 
add(p, BorderLayout.SOUTH) ; 
} 
public static void main(String args[]) { 
Show.inFrame(new Trees(),200, 500); 


} 
LI] i 








最 重要 的 类 束 是 分 支 ， 它 是 一 个 工具 ， 用 来 获取 一 个 字符 串 数 组 并 为 第 
一 个 字符 串 建立 一 个 DefaultMutableTreeNode 作 为 根 ， 其 余 在 数组 中 的 
字符 串 作 为 叶 。 然 后 node() 方 法 被 调用 以 产生 “分 支 ”? 的 根 。 树 状 物 类 包 


括 一 个 来 目 被 制造 的 分 文 的 二 维 字 符 串 数组 ， 以 及 用 来 统计 数组 的 一 个 
静态 中 汤 i。DefaultMutableTreeNode 对 象 控 制 这 个 结 节 ， 但 在 屏幕 上 表 
示 的 是 被 JTree 和 它 的 相关 (DefaultTreeModel) 模式 所 控制 。 注 意 当 
JTree 被 增加 到 程序 片 时 ， 它 被 封装 到 JScrollPane 中 te Cp HE 
供 的 目 动 深 动 。 


JTree 通 过 它 自 己 的 模型 来 控制 。 当 我 们 修改 这 个 模型 时 ， 模 型 产生 一 
个 事件 ， 寻 致 JTree 对 可 以 看 见 的 树 状 物 完 成 任何 必要 的 升级 。 在 initO 
中 ， 模 型 由 调用 getModel0 方 法 所 捕捉 。 当 按钮 被 按 下 时 ， 一 个 新 的 分 
文 被 创建 了 。 然 后 ， 当 前 选择 的 组 件 被 找到 《〈 如 果 没 有 选择 就 是 根 ) 并 
的 insertNodeInto0) 方 法 做 所 有 的 改变 树 状 物 和 导致 它 升级 的 工 


大 多 数 的 时 候 ， 就 像 上 面 的 例子 一 样 ， 程 序 将 给 我 们 在 树 状 物 中 所 需要 
的 一 切 。 不 过 ， 树 状 物 拥有 力量 去 做 我 们 能 够 想像 到 的 任何 事 一 一 在 上 
面 的 例子 中 我 们 到 处 都 可 看 到 “default〈 默 认 ) ”字样 ， 我 们 可 以 取代 我 
们 自己 的 类 来 获取 不 同 的 动作 。 但 请 注意 : 几乎 所 有 这 些 类 都 有 一 个 具 
接口 ， 因 此 我 们 可 以 花 一 些 时 间 努 力 去 理解 这 些 错综复杂 的 树 状 











13.19.14 表格 


和 树 状 物 一 样 ， 表 格 在 Swing 相当 的 庞大 和 强大 。 它 们 最 初 有 意 被 设计 

成 以 Java 数 据 库 连结 (JDBC， 在 15 章 有 介绍 ) 为 媒介 的 “网 格 ” 数 据 库 接 
口 ， 并 且 因 此 它们 拥有 的 巨大 的 灵活 性 ， 使 我 们 不 再 感到 复杂 。 无 疑 ， 

这 是 足以 成 为 成 熟 的 电子 数据 表 的 基础 条 件 而 且 可 能 为 整 本 书 提供 很 好 
但 是 ， 如 果 我 们 理解 这 个 的 基础 条 件 ， 它 同样 可 能 创建 相关 的 
简单 的 Jtable。 


JTable 控 制 数据 的 显示 方式 ， 但 TableModel 控 制 它 自己 的 数据 。 因 此 在 
我 们 创建 JTable 前 ， 应 先 创建 一 个 TableModel。 我 们 可 以 全 部 地 执行 
TableModel 接 口 ， 但 它 通常 从 helper 类 的 AbstractTableModel 处 简单 地 继 
JK: 


//: Table.java 

















// Simple demonstration of JTable 


package c13.Swing; 
import java.awt.*; 
import java.awt.event.*; 
import javax.swing.*; 
import javax.Swing.table.*; 
import javax.swing.event.*; 
// The TableModel controls all the data: 
class DataModel extends AbstractTableModel { 
Object[][] data = { 
{"one", "two", "three", "four"}, 
{"five", "six", "seven", "eight"}, 
{"nine", "ten", "eleven", "twelve"}, 
}; 
// Prints data when table changes: 
class TML implements TableModelListener { 
public void tableChanged(TableModelEvent e) { 
for(int i = 0; i < data.length; i++) { 
for(int j = 0; j < data[O].length; j++) 
System.out.print(data[i]J[j] + " "); 


System.out.printin(); 


DataModel() { 
addTableModelListener(new TML()); 

} 

public int getColumnCount() { 
return data[0].length; 

} 

public int getRowCount() { 
return data.length; 

} 

public Object getValueAt(int row, int col) { 
return data[row][col]; 

} 

public void 

setValueAt(Object val, int row, int col) { 
data[row][col] = val; 
// Indicate the change has happened: 
fireTableDataChanged(); 

} 

public boolean 

isCellEditable(int row, int col) { 


return true; 


public class Table extends JPanel { 

public Table() { 
setLayout(new BorderLayout()); 
JTable table = new JTable(new DataModel()); 
JScrollPane scrollpane = 

JTable.createScrollPaneForTable(table); 

add(scrollpane, BorderLayout.CENTER); 

} 

public static void main(String args[]) { 
Show. inFrame(new Table(),200, 200); 


} 
} ///:~ 


DateModel 包 括 一 组 数据 ， 但 我 们 同样 能 从 其 它 的 地 方 得 到 数据 ， 例 如 
从 数据 库 中 。 构 建 器 增加 了 一 个 TableModelListener 用 来 在 每 次 表格 被 改 
变 后 打印 数组 。 剩 下 的 方法 都 遵循 Bean 的 命名 规则 ， 并 且 当 JTable 需 要 
在 DateModel 中 显示 信息 时 调用 。AbstractTableModel 提 供 了 默认 的 
setValueAtO0 和 isCellEditable() 方 法 以 防止 修改 这 些 数据 ， 因 此 如 果 我 们 
想 修 改 这 些 数据 ， 就 必须 过 载 这 些 方法 。 


一 旦 我 们 拥有 一 个 TableModel， 我 们 只 需要 将 它 分 配给 JTable 构 建 右 即 
可 。 所 有 有 关 显 示 ， 编 辑 和 更 新 的 详细 资料 将 为 我 们 处 理 。 注 意 这 个 程 
序 例子 同样 将 JTable 放 置 在 JScrollPane 中 ， 这 是 因为 JScrollPane 需 要 一 个 
特殊 的 JTable 方 法 。 


13.19.15 卡片 式 对 话 框 
在 本 章 的 前 部 ， 回 我 们 介绍 了 老式 的 CardLayout， 并 且 注 意 到 我 们 怎样 


去 管理 我 们 所 有 的 卡片 开关 。 有 趣 的 是 ， 有 人 现在 认为 这 是 一 种 不 错 的 
设计 。 幸 运 的 是 ，Swing 用 JTabbedPane 对 它 进行 了 修补 ， 由 











JITabbedPane 来 处 理 这 些 卡 片 ， 开 关 和 其 它 的 任何 事物 。 对 比 CardLayout 
和 JTabbedPane， 我 们 会 发 现 惊人 的 差异 。 


下 面 的 程序 例子 十 分 的 有 趣 ， 因 为 它 利 用 了 前 面 例 子 的 设计 。 它 们 都 是 
做 为 JPanel 的 衍生 物 来 构建 的 ， 因 此 这 个 程序 将 安放 前 面 的 每 个 例子 到 
它 自 己 在 JTabbedPane 的 窗 格 中 。 我 们 会 看 到 利用 RTTI 制 造 的 程序 十 分 
的 小 巧 精致 : 


//: Tabbed.java 


// Using tabbed panes 

package c13.swing; 

import java.awt.*; 

import javax.swing.*; 

import javax.swing.border.*; 

public class Tabbed extends JPanel { 

static Object[][] q = { 

"Felix", Borders.class }, 
"The Professor", Buttons.class }, 
"Rock Bottom", ButtonGroups.class }, 


"Theodore", Faces.class }, 


{ 

{ 

{ 

{ 

{ "Simon", Menus.class }, 
{ "Alvin", Popup.class }, 

{ "Tom", ListCombo.class }, 
{ "Jerry", Progress.class }, 
{ 


"Bugs", Trees.class }, 


}; 


{ "Daffy", Table.class }, 


static JPanel makePanel(Class c) { 


i 


String title = c.getName(); 

title = title.substring( 
title.lastIndexOf('.') + 1); 

JPanel sp = null; 

try { 
sp = (JPanel)c.newInstance(); 

} catch(Exception e) { 
System.out.println(e); 

} 

sp.setBorder(new TitledBorder(title) ); 


return sp; 


public Tabbed() { 


setLayout(new BorderLayout()); 
JTabbedPane tabbed = new JTabbedPane(); 
for(int i = 0; i < q.length; i++) 
tabbed.addTab((String)q[i][0], 
makePanel((Class)q[i][1])); 
add(tabbed, BorderLayout.CENTER) ; 


tabbed. setSelectedIndex(q.length/2); 


} 
public static void main(String args[]) { 


Show.inFrame(new Tabbed(), 460,350); 


} 
} ///:~ 


再 者 ， 我 们 可 以 注意 到 使 用 的 数组 构造 式样 : 第 一 个 元 素 是 被 置 放 在 卡 
片上 的 String， 第 二 个 元 素 是 将 被 显示 在 对 应 窗 格 上 JPanel 类 。 在 
Tabbed() 构 建 嚣 里， 我们 可 以 看 到 两 个 重要 的 JTabbedPane 方 法 被 使 用 : 
addTab() 插 入 一 个 新 的 窗 格 ，setSelectedIndex() 选 择 一 个 窗 格 并 从 它 开 
始 。“ 一 个 在 中 间 被 选中 的 窗 格 证 明 我 们 不 必 从 第 一 个 窗 格 开始 ) 。 


当 我 们 调用 addTab() 方 法 时 ， 我 们 为 它 提供 卡片 的 String 和 一 些 组 件 (也 
就 是 说 ， 一 个 AWT 组 件 ， 而 不 是 一 个 来 自 AWT 的 JComponent)。 这 个 
组 件 会 被 显示 在 窗 格 中 。 一 旦 我 们 这 样 做 了 ， 自 然而 然 的 就 不 需要 更 多 
管理 了 JTabbedPane 会 为 我 们 处 理 其 它 的 任何 事 。 


makePanel() 方 法 获取 我 们 想 创 建 的 类 Class 对 象 和 用 newInstance() 去 创建 
并 造型 为 JPanel (当然 ， 假 定 那 些 类 是 必须 从 JPanel 继 承 才 能 增加 的 
类 ， 除 非 在 这 一 节 中 为 程序 例子 的 结构 所 使 用 ) 。 它 增加 了 一 个 包括 类 
名 并 返回 结果 的 TiledBorder， 以 作为 一 个 JPanel 在 addTab() 被 使 用 。 


当 我 们 运行 程序 时 ， 我 们 会 发 现 如 果 卡 片 太 多 ， 填 满 了 一 行 ， 
JTabbedPane 自 动 地 将 它们 堆积 起 来 。 














13.19.16 Swing 消 息 框 


开 窗 的 环境 通常 包含 一 个 标准 的 信息 框 集 ， 人 允许 我 们 很 快 传递 消息 给 用 
户 或 者 从 用 户 那里 捕 提 消息 。 在 Swing 里 ， 这 些 信息 窗 被 包含 在 
JOptionPane 里 的 。 我 们 有 一 些 不 同 的 可 能 实现 的 事件 (有 一 些 十 分 复 
杂 ) ， 但 有 一 点 ， 我 们 必须 尽 可 能 的 利用 static 
JOptionPane.showMessageDialog() 和 JOptionPane.showConfirmDialog()77 
法 ， 调 用 消息 对 话 框 和 确认 对 话 框 。 


13.19.17 Swing 更 多 的 知识 

















这 一 节 意 味 着 唯一 同 我 们 介绍 的 是 Swing 的 强大 力量 和 我 们 的 着 手 处 ， 
因此 我 们 能 注意 到 通过 库 ， 我 们 会 感觉 到 我 们 的 方法 何等 的 简单 。 到 目 
前 为 止 ， 我 们 已 看 到 的 可 能 足够 满足 我 们 UI 设 计 需 要 的 一 部 分 。 不 过 ， 
这 里 有 许多 有 关 Swing 额 外 的 情况 一 一 它 有 意 成 为 一 全 功能 的 UI 设 计 工 
具 箱 。 如 果 我 们 没有 发 现 我 们 所 需要 的 ， 请 到 SUN 公 司 的 在 线 文 件 中 去 
查找 ， 并 搜索 WEB。 这 个 方法 几乎 可 以 完成 我 们 能 想到 的 任何 事 。 

本 节 中 没有 涉及 的 一 些 要 点 : 

四 更 多 特殊 的 组 件 ， 例 如 
JColorChooser,JFileChooser,JPasswordField,JHTMLPane 〈 完 成 简单 的 
HTML 格 式 化 和 显示 ) 以 及 JTextPane (一 个 支持 格式 化 ， 字 人 处理 和 图 像 
的 文字 编辑 器 ) 。 它 们 都 非常 易 用 。 


@Swing 的 新 的 事件 类 型 。 在 一 些 方法 中 ， 它 们 看 起 来 像 违例 :类 型 非常 
的 重要 ， 名 字 可 以 被 用 来 表示 除了 它们 目 己 之 外 的 任何 事物 。 


新 的 布局 管理 : Springs & Struts 以 及 BoxLayout 


一 个 间 隅 物 式 的 分 多 条 ， 人 允许 我 们 动态 地 处 理 其 它 组 件 的 
Eo 


国 JLayeredPane 和 JInternalFrame 被 一 起 用 来 在 当前 帧 中 创建 子 帧 ， 以 产 
生 多 文件 接口 (MDI) 应 用 程序 。 


四 可 插入 的 外 观 和 效果 ， 因 此 我 们 可 以 编写 单个 的 程序 可 以 像 期 望 的 那 
样 动态 地 适合 不 同 的 平台 和 操作 系统 。 


mE Oth 

mJToolbar API 提 供 的 可 拖 动 的 浮动 工具 条 。 

四 双 绥 存 和 为 平整 屏幕 重新 画 线 的 自动 重 画 批 次 。 
mY EE SC HF 

mite SCH 

















13.20 总 结 


对 于 AWT 而 言 ，Java 1.1 到 Java 1.2 最 大 的 改变 就 是 Java 中 所 有 的 库 。 
Java 1.0 版 的 AWT 曾 作为 目前 见 过 的 最 烛 糕 的 一 个 设计 被 彻底 地 批评 ， 
并 且 当 它 人 允许 我 们 在 创建 小 巧 精致 的 程序 时 ， 产 生 的 GUI“ 在 所 有 的 平 
台 上 都 同样 的 平庸”*”。 它 与 在 特殊 平台 上 本 地 应 用 程序 开发 工具 相 比 也 
是 受到 限制 的 ， 笨 拙 的 并 且 也 是 不 友好 的 。 当 Java” ”1.1 版 纳入 新 的 事件 
模型 和 Java Beans 时 ， 平 台 被 设置 现在 它 可 以 被 拖 放 到 可 视 化 的 应 
用 程序 构建 工具 中 ， 创建 GUI 组 件 。 另 外 ， 事 件 模型 的 设计 和 Bean 无 疑 
对 轻松 的 编程 和 可 维护 的 代码 都 非常 的 在 意 〈 这 些 在 Java 1.0 AWT 中 不 
那么 的 明显 ) 。 但 直至 GUI 组 件 一 下 C/Swing 类 一 显示 工作 结束 它 才 这 
I 交叉 平台 GUI 编 程 可 以 变 成 一 种 有 教育 意义 


现在 ， 唯 一 的 情况 是 缺乏 应 用 程序 构建 工具 ， 并 且 这 就 是 真正 的 变革 的 
存在 之 处 。 微 软 的 Visual Basic 和 Visual C++ 需要 它们 的 应 用 程序 构建 工 
具 ， 同 样 的 是 Borland 的 Deljphi 和 C++ 构建 问 。 如 果 我 们 需要 应 用 程序 构 
建 工 具 变 得 更 好 ， 我 们 不 得 不 交叉 我 们 的 指针 并 且 和 希望 自动 授权 机 会 给 
我 们 所 需要 的 。Java 是 一 个 开放 的 环境 ， 因 此 不 但 考虑 到 同 其 它 的 应 用 
程序 构建 环境 竞争 ， 而 且 Java 还 促进 它们 的 发 展 。 这 些 工具 被 认真 地 使 
用 ， 它 们 必须 支持 Java ”Beans。 这 意味 着 一 个 平等 的 应 用 领域 ， 如 果 一 
个 更 好 的 应 用 程序 构建 工具 出 现 ， 我 们 不 需要 去 约束 它 就 可 以 使 用 
我 们 可 以 采用 并 移动 到 新 的 工具 上 工作 即 可 ， 这 会 提高 我 们 的 工作 效 
率 。 这 种 竞争 的 环境 对 应 用 程序 构建 工具 来 说 从 未 出 现 过 ， 这 种 竞争 能 
真正 提高 程序 设计 者 的 工作 效率 。 



































13.21 练习 


(1) 创 建 一 个 有 文字 字段 和 三 个 按钮 的 程序 片 。 当 我 们 按 下 每 个 按钮 时 ， 
使 不 同 的 文字 显示 在 文字 段 中 。 


(2) 增 加 一 个 复 选 框 到 练习 1 创建 的 程序 中 ， 捕 捉 事 件 ， 并 插入 不 同 的 文 
PR he 

(3) 创 建 一 个 程序 片 并 增加 所 有 导致 action() 被 调用 的 组 件 ， 然 后 捕捉 他 
们 的 事件 并 在 文字 字段 中 为 每 个 组 件 显示 一 个 特定 的 消 居 。 


(4) 增 加 可 以 被 handleEvent() 方 法 测试 事件 的 组 件 到 练习 3 中 。 过 载 
handleEvent() 并 在 文字 字段 中 为 每 个 组 件 显 示 特 定 的 消 旦 。 


(5) 创 建 一 个 有 一 个 按钮 和 一 个 TextField 的 程序 片 。 编 写 一 个 
handleEvent()， 以 便 如 果 按 钮 有 焦点 ， 输 入 字符 到 将 显示 的 TextField 
中 。 

















(6) 创 建 一 个 应 用 程序 并 将 本 章 所 有 的 组 件 增加 主要 的 帧 ， 包 括 菜单 和 对 


话 框 。 


T 以 便 字 母 在 2 中 保持 输入 时 的 样子 ， 取 代 自 动 变 
BRS 2 








(8) 修 改 CardLayonutl.java 以 便 它 使 用 Java 1.1 的 事件 模型 。 


(9) 增 加 Frog.class 到 本 章 出 现 的 清单 文件 中 并 运行 jar 以 创建 一 个 包括 Frog 
和 BangBean 的 JAR 文 件 。 现 在 从 SUN 公 司 处 下 载 并 安装 BDK 或 者 使 用 我 
们 自己 的 可 激活 Bean 的 程序 构建 工具 并 增加 JAR 文 件 到 我 们 的 环境 中 ， 
因此 我 们 可 以 测试 两 个 Bean。 


(10) 创 建 我 们 自己 的 包括 两 个 属性 ， 一 个 布尔 值 为 “on”， 另 一 个 为 整 
型 level”， 称 为 Valve 的 Java Bean。 创 建 一 个 清单 文件 ， 利 用 jar 打 包 我 
们 的 Bean， 然 后 读 入 它 到 beanbox 或 到 我 们 自己 的 激活 程序 构建 工具 

里 ， 因 此 我 们 可 以 测试 它 。 


(11) 修 改 Menus.java， 以 便 它 处 理 多 级 菜单 。 这 要 假设 读者 已 经 熟悉 了 


HTML 的 基础 知识 。 但 那些 东西 并 不 难 理解 ， 而 且 有 一 些 书 和 资料 可 供 


B 


第 14 章 多 线程 


利用 对 象 ， 可 将 一 个 程序 分 割 成 相互 独立 的 区 域 。 我 们 通常 也 需要 将 一 
个 程序 转换 成 多 个 独立 运行 的 子 任务 。 


象 这 样 的 每 个 子 任务 都 叫 作 一 个 “线程 ”〈Thread) 。 编 写 程序 时 ， 可 将 
每 个 线程 都 想象 成 独立 运行 ， 而 且 都 有 目 己 的 专用 CPU。 一 些 基 础 机 制 
实际 会 为 我 们 自动 分 割 CPU 的 时 间 。 我 们 通常 不 必 关 心 这 些 细 市 问题 ， 
所 以 多 线程 的 代码 编写 是 相当 简便 的 。 


这 时 理解 一 些 定义 对 以 后 的 学 习 锋 有 帮助 。“ 进 程 * 是 指 一 种 “ 目 包容 ”的 
运行 程序 ， 有 上 自己 的 地 址 空间 。“ 多 任务 ”操作 系统 能 同时 运行 多 个 进程 
GEF) 一 一 但 实际 是 由 于 CPU 分 时 机 制 的 作用 ， 使 每 个 进程 都 能 循环 
获得 自己 的 CPU 时 间 片 。 但 由 于 轮换 速度 非常 快 ， 使 得 所 有 程序 好 象 是 
在 “同时 ?运行 一 样 。“ 线 程 ? 是 进程 内 部 单一 的 一 个 顺序 控制 流 。 因 此 ， 
一 个 进程 可 能 容纳 了 多 个 同时 执行 的 线程 。 


多 线程 的 应 用 范围 很 广 。 但 在 一 般 情 况 下 ， 程 序 的 一 些 部 分 同 特定 的 事 
件 或 资源 联系 在 一 起 ， 同 时 又 不 想 为 它 而 暂停 程序 其 他 部 分 的 执行 。 这 
样 一 来 ， 就 可 考虑 创建 一 个 线程 ， 令 其 与 那个 事件 或 资源 关联 到 一 起 ， 
并 让 它 独 立 于 主 程序 运行 。 一 个 很 好 的 例子 便 是 “Quit* 或 “退出 ”按钮 

我 们 并 不 希望 在 程序 的 每 一 部 分 代码 中 都 轮 询 这 个 按钮 ， 同 时 又 希 
望 该 按钮 能 及 时 地 作出 啊 应 使 程序 看 起 来 似乎 经 常 都 在 轮 询 它 ) 。 事 
实 上 ， 多 线程 最 主要 的 一 个 用 途 束 是 构建 一 个 “反应 灵敏 ”的 用 户 界 面 。 









































14.1 反应 灵敏 的 用 户 界 面 


作为 我 们 的 起 点 ， 请 思考 一 个 需要 执行 某 些 CPU 密集 型 计算 的 程序 。 由 
于 CPU“ 人 全心全意? 为 那些 计算 服务 ， 所 以 对 用 户 的 输入 十 分 迟钝 ， 几 乎 
没有 什么 反应 。 在 这 里 ， 我 们 用 一 个 合成 的 applet/application GEF A / 
应 用 程序 ) 来 简单 显示 出 一 个 计数 器 的 结果 : 


//: Counter1.java 








// A non-responsive user interface 
package c14; 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
public class Counteri extends Applet { 
private int count = 0; 
private Button 
onoff = new Button("Toggle"), 
start = new Button("Start"); 
private TextField t = new TextField(10); 
private boolean runFlag = true; 
public void init() { 
add(t); 
start.addActionListener(new StartL()); 


add(start); 


onoff ,addActionListener(new OnOffL()); 
add(onOff); 
} 
public void go() { 
while (true) { 
try { 
Thread.currentThread().sleep(100) ; 
} catch (InterruptedException e){} 
if(runFlag) 


t.setText(Integer.toString(count++) ); 


} 


class StartL implements ActionListener { 


public void actionPerformed(ActionEvent e) { 


go(); 


} 


class OnoffL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


runFlag = !runFlag; 


} 


public static void main(String[] args) { 


Counter1 applet = new Counter1(); 
Frame aFrame = new Frame("Counteri"); 
aFrame.addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
} ///:~ 


在 这 个 程序 中 ，AWT 和 程序 片 代码 都 应 是 大 家 熟悉 的 ， 第 13 章 对 此 已 
有 很 详细 的 交待 。go0) 方 法 正 是 程序 全 心 全 意 服 务 的 对 待 : 将 当前 的 
count (计数 ) 值 置 入 TextField〈 文 本 字段 ) tf， 然 后 使 count 增 值 。 


go0 内 的 部 分 无 限 循环 是 调用 sleepO0。sleep0 必 须 同 一 个 Thread (FE) 
对 象 关联 到 一 起 ， 而 且 似 乎 每 个 应 用 程序 都 有 部 分 线程 同 它 关联 (事实 
上 ，Java 本 身 就 是 建立 在 线程 基础 上 的 ， 肯 定 有 一 些 线程 会 伴随 我 们 写 
的 应 用 一 起 运行 )。 所 以 无 论 我 们 是 否 明 确 使 用 了 线程 ， 都 可 利用 
Thread.currentThreadO 产 生 由 程序 使 用 的 当前 线程 ， 然 后 为 那个 线程 调 
用 sleep()。 注 意 ，Thread.currentThread() 是 Thread 类 的 一 个 静态 方法 。 








注意 sleepO0 可 能 “ 掷 ?出 一 个 InterruptException CPER) 一 一 尽管 产生 


这 样 的 违例 被 认为 是 中 止 线程 的 一 种 “恶意 手段， 而 且 应 该 尽 可 能 地 杜 
绝 这 一 做 法 。 再 次 提醒 大 家 ， 违 例 是 为 异常 情况 而 产生 的 ， 而 不 是 为 了 
正常 的 控制 流 。 在 这 里 包含 了 对 一 个 “睡眠 ”线程 的 中 断 ， 以 文 持 未 来 的 


一 种 语言 特性 。 


一 旦 按 下 start 按 钮 ， 就 会 调用 go()。 研 究 一 下 go()， 你 可 能 会 很 自然 地 
( 束 象 我 一 样 ) 认 为 它 该 支持 多 线程 ， 因 为 它 会 进入 “睡眠 ”状态 。 也 就 
是 说 ， 尺 管 方法 本 身 “ 睡 着 ”了 ，CPU 仍 然 应 该 忙于 监视 其 他 按钮 “ 按 

下 ?事件 。 但 有 一 个 问题 ， 那 融 是 go0 是 永远 不 会 返回 的 ， 因 为 它 被 设计 
成 一 个 无 限 循环 。 这 意味 着 actionPerformed() 根 本 不 会 返回 。 由 于 在 第 
一 个 按键 以 后 便 陷入 actionPerformed() 中 ， 所 以 程序 不 能 再 对 其 他 任何 
事件 进行 控制 (如 果 想 出 来 ， 必 须 以 某 种 方式 “ 杀 死 ”进程 最 简便 的 
方式 就 是 在 控制 台 窗 口 按 Ctrl 十 C 键 ) 。 


这 里 最 基本 的 问题 是 go0 需 要 继续 执行 目 己 的 操作 ， 而 与 此 同时 ， 它 也 
需要 人 返回， 以 便 actionPerformed() 能 够 完成 ， 而 且 用 户 界 面 也 能 继续 啊 
应 用 户 的 操作 。 但 对 象 go0 这 样 的 传统 方法 来 说 ， 它 却 不 能 在 继续 的 同 
时 将 控制 权 返 回 给 程序 的 其 他 部 分 。 这 上 听 起 来 似乎 是 一 件 不 可 能 做 到 的 
事情 ， 就 象 CPU 必 须 同 时 位 于 两 个 地 方 一 样 ， 但 线程 可 以 解雇 一 

YW. “线程 模型 ”〈 以 及 Java 中 的 编程 文 持 ) 是 一 种 程序 编写 规范 ， 可 在 
单独 一 个 程序 里 实现 几 个 操作 的 同时 进行 。 根 据 这 一 机 制 ，CPU 可 为 每 
个 线程 都 分 配 上 自己 的 一 部 分 时 间 。 每 个 线程 都 “感觉 "自己 好 象 拥有 整个 
CPU， 但 CPU 的 计算 时 间 实 际 却 是 在 所 有 线程 间 分 摊 的 。 


线程 机 制 多 少 降 低 了 一 些 计算 效率 ， 但 无 论 程 序 的 设计 ， 资 源 的 均衡 ， 
还 古 用 户 操作 的 方便 性 ， 都 从 中 获得 了 巨大 的 利益 。 综 合 考虑 ， 这 一 机 
制 是 非常 有 价值 的 。 当 然 ， 如 果 本 来 束 安 装 了 多 块 CPU， 那 么 操作 系统 
能 够 自行 决定 为 不 同 的 CPU 分 配 哪些 线程 ， 程 序 的 总 体 运行 速度 也 会 变 
得 更 快 ( 所 有 这 些 部 要 求 操作 系统 以 及 应 用 程序 的 支持 ) 。 多 线程 和 多 
任务 是 充分 发 挥 多 处 理 机 系统 能 力 的 一 种 最 有 效 的 方式 。 


14.1.1 从 线程 继承 


为 创建 一 个 线程 ， 最 简单 的 方法 就 是 从 Thread 类 继承 。 这 个 类 包含 了 创 
建 和 运行 线程 所 需 的 一 切 东 西 。Thread 最 重要 的 方法 是 run()。 但 为 了 使 
用 run0)， 必 须 对 其 进行 过 载 或 者 履 盖 ， 使 其 能 充分 按 上 自己 的 吟 只 行 事 。 
ae run0 属 于 那些 会 与 程序 中 的 其 他 线程 “并 发 ?或 < 同时 ?执行 的 代 



































下 面 这 个 例子 可 创建 任意 数量 的 线程 ， 并 通过 为 每 个 线程 分 配 一 个 独 一 
无 二 的 编写 (由 一 个 静态 变量 产生 ) ， 从 而 对 不 同 的 线程 进行 跟踪 。 
Thread 的 run() 方 法 在 这 里 得 到 了 禾 产 ， 每 通过 一 次 循环 ， 计 数 就 减 1 
一 一 计数 为 0 时 则 完成 循环 (此 时 一 旦 返回 run()， 线 程 束 中 止 运行 〉。 


//: SimpleThread. java 





// Very simple Threading example 
public class SimpleThread extends Thread { 
private int countDown = 5; 
private int threadNumber; 
private static int threadCount = 0; 
public SimpleThread() { 
threadNumber = ++threadCount; 
System.out.printin( "Making " + threadNumber ); 
} 
public void run() { 
while(true) { 
System.out.println("Thread " + 
threadNumber + "(" + countDown + ")"); 


if(--countDown == 0) return; 


} 


public static void main(String[] args) { 


for(int i = 0; i < 5; i++) 


new SimpleThread().start(); 


System.out.printin("All Threads Started"); 


A f/f 





直 持 续 到 线程 不 再 
要 为 止 。 因 此 ， 我 们 必须 规定 特定 的 条 件 ， 以 便 中 断 并 退出 这 个 循环 
《或 者 在 上 述 的 例子 中 ， 简 单 地 从 run0 返 回 即 可 ) 。run0 通 常 采 用 一 种 

无 限 循环 的 形式 。 也 束 是 说 ， 通 过 阻止 外 部 发 出 对 线程 的 stop0 或 者 

destroyO 调 用 ， 它 会 永远 运行 下 去 〈 直 到 程序 完成 ) 。 


在 main() 中 ， 可 看 到 创建 并 运行 了 大 量 线程 。Thread 包 含 了 一 个 特殊 的 
方法 ， 叫 作 start0， 它 的 作用 是 对 线程 进行 特殊 的 初始 化 ， 然 后 调用 

run0。 所 以 整个 步骤 包括 : 调用 构建 器 来 构建 对 象 ， 然 后 用 startO 配 置 
线程 ， 再 调用 run0。 如 果 不 调 用 startO) 如 果 适 当 的 话 ， 可 在 构建 器 














下 面 是 该 程序 茶 一 次 运行 的 输出 《注意 每 次 运行 都 会 不 同 ) : 


Making 1 


N 


Making 
Making 3 
Making 4 
Making 5 
Thread 1(5) 
Thread 1(4) 
Thread 1(3) 


Thread 1(2) 


Thread 2(5) 
Thread 2(4) 
Thread 2(3) 
Thread 2(2) 
Thread 2(1) 
Thread 1(1) 
All Threads Started 
Thread 3(5) 
Thread 4(5) 
Thread 4(4) 
Thread 4(3) 
Thread 4(2) 
Thread 4(1) 
Thread 5(5) 
Thread 5(4) 
Thread 5(3) 
Thread 5(2) 
Thread 5(1) 
Thread 3(4) 
Thread 3(3) 
Thread 3(2) 


Thread 3(1) 


可 注意 到 这 个 例子 中 到 处 都 调用 了 sleep0， 然 而 输出 结果 指出 每 个 线程 
都 获得 了 属于 自己 的 那 一 部 分 CPU 执行 时 间 。 从 中 可 以 看 出 ， 尽 管 
sleep0) 依 赖 一 个 线程 的 存在 来 执行 ， 但 却 与 允许 或 茶 止 线程 无 天 。 它 只 
不 过 是 为 一 个 不 同 的 方法 而 已 。 


亦 可 看 出 线程 并 不 是 按 它们 创建 时 的 顺序 运行 的 。 事 实 上 ，CPU 处 理 一 
个 现 有 线程 集 的 顺序 是 不 确定 的 一 一 除非 我 们 杀 目 介入 ， 并 用 Thread 的 
setPriority() 方 法 调整 它们 的 优先 级 。 


main(0) 创 建 Thread 对 象 时 ， 它 并 未 捕获 任何 一 个 对 象 的 句柄 。 普 通 对 象 
对 于 垃圾 收集 来 说 是 一 种 “公平 竞赛 ”， 但 线程 却 并 非 如 此 。 每 个 线程 都 
会 “注册 ” 目 己 ， 所 以 某 处 实际 存在 着 对 它 的 一 个 引用 。 这 样 一 来 ， 垃 圾 
Wee ae te R ATE E AT? T o 

14.1.2 针对 用 户 界面 的 多 线程 

现在 ， 我 们 也 许 能 用 一 个 线程 解决 在 Counter1l.java 中 出 现 的 问题 。 采 用 
的 一 个 技巧 便 是 在 一 个 线程 的 run0) 方 法 中 放置 “ 子 任务 ” 亦 即 位 于 
go0 内 的 循环 。 一 旦 用 户 按 下 Start 按 钮 ， 线 程 就 会 司 动 ， 但 马上 结束 线 
程 的 创建 。 这 样 一 来 ， 尽 管线 程 仍 在 运行 ， 但 程序 的 主要 工作 却 能 得 以 
继续 〈 等 候 并 啊 应 用 户 界面 的 事件 ) 。 下 面 是 具体 的 代码 : 


//: Counter2.java 






































// A responsive user interface with threads 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
class SeparateSubTask extends Thread { 
private int count = 0; 
private Counter2 c2; 


private boolean runFlag = true; 


public SeparateSubTask(Counter2 c2) { 
this.c2 = c2; 
start(); 
} 
public void invertFlag() { runFlag = !runFlag;} 
public void run() { 
while (true) { 
try { 
sleep(100); 
} catch (InterruptedException e){} 
if (runFlag) 


c2.t.setText(Integer.toString(count++) ); 


} 


public class Counter2 extends Applet { 
TextField t = new TextField(10); 
private SeparateSubTask sp = null; 
private Button 
onOff = new Button("Toggle"), 
start = new Button("Start"); 
public void init() { 


add(t); 


start.addActionListener(new StartL()); 
add(start); 
onOff.addActionListener(new OnOffL()); 
add(onOff); 

} 

class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 

if(sp == null) 


sp = new SeparateSubTask(Counter2.this); 


} 


class OnoffL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(sp != null) 


sp.invertFlag(); 


} 


public static void main(String[] args) { 
Counter2 applet = new Counter2(); 
Frame aFrame = new Frame("Counter2"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 


public void windowClosing(WindowEvent e) 


System.exit(0); 
} 

}); 
aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 


} 
} ///:~ 


现在 ，Counter2 变 成 了 一 个 相当 直接 的 程序 ， 它 的 唯一 任务 就 是 设置 并 
管理 用 户 界 面 。 但 假 知 用 户 现 在 按 下 Start 按 钮 ， 却 不 会 真正 调用 一 个 方 
法 。 此 时 不 是 创建 类 的 一 个 线程 ， 而 是 创建 SeparateSubTask， 然 后 继续 
Counter2 事 件 循环 。 注 意 此 时 会 保存 SeparateSubTask 的 句柄 ， 以 便 我 们 
按 下 onOff 按 钮 的 时 候 ， 能 正常 地 切换 位 于 SeparateSubTask 内 部 的 
runFlag (运行 标志 ) 。 随 后 那个 线程 便 可 启动 ( 当 它 看 到 标志 的 时 
候 ) ， 然 后 将 自己 中 止 〈 亦 可 将 SeparateSubTask 设 为 一 个 内 部 类 来 达到 
这 一 目的 ) 。 


SeparateSubTask 类 是 对 Thread 的 一 个 简单 扩展 ， 它 融 有 一 个 构建 器 (其 
中 保存 了 Counter2 句 柄 ， 然 后 通过 调用 start0) 来 运行 线程 ) 以 及 一 个 rung) 
一 一 本 质 上 包含 了 Counterl.java 的 go0 内 的 代码 。 由 于 SeparateSubTask 
知道 自己 容纳 了 指 回 一 个 Counter2 的 句柄 ， 所 以 能 够 在 需要 的 时 候 介 
入 ， 并 访问 Counter2 的 TestField〈 文 本 字段 ) 。 


按 下 onOff 按 钮 ， 几 乎 立即 能 得 到 正确 的 响应 。 当 然 ， 这 个 啊 应 其 实 并 
不 是 “立即 ?发 生 的 ， 它 毕竟 和 那 种 由 “中 断 ” 驱 动 的 系统 不 同 。 只 有 线程 
拥有 CPU 的 执行 时 间 ， 并 注意 到 标记 已 发 生 改变 ， 计 数 需 才 会 俘 止 。 


1. 用 内 部 类 改善 代码 











下 面 说 说 题 外 话 ， 请 大 家 注意 一 下 SeparateSubTask 和 Counter2 类 之 间 发 
生 的 结合 行为 。SeparateSubTask 同 Counter2“ 亲 密 *” 地 结合 到 了 一 起 一 一 
它 必须 持 有 指 癌 自己“ 父 "Counter2 对 象 的 一 个 句柄 ， 以 便 自 己 能 回调 和 
操纵 它 。 但 两 个 类 并 不 是 真 的 合并 为 单独 一 个 类 【尽管 在 下 一 节 中 ， 我 
们 会 讲 到 Java 确 实 提供 了 合并 它们 的 方法 ) ， 因 为 它们 各 自 做 的 是 不 同 
的 事情 ， 而 且 是 在 不 同 的 时 间 创 建 的 。 但 不 管 怎样 ， 它 们 依然 紧密 地 结 
合 到 一 起 《更 准确 地 说 ， 应 该 叫 “ 联 合 ”) ， 所 以 使 程序 代码 多 少 显 得 有 
n 在 这 种 情况 下 ， 一 个 内 部 类 可 以 显著 改善 代码 的 “可 读 性 ”和 执 
行 效率 : 


//: Counter2i.java 





// Counter2 using an inner class for the thread 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
public class Counter2i extends Applet { 
private class SeparateSubTask extends Thread { 
int count = 0; 
boolean runFlag = true; 
SeparateSubTask() { start(); } 
public void run() { 
while (true) { 
try { 
sleep(100); 
} catch (InterruptedException e){} 


if(runFlag) 


t.setText(Integer.toString(count++) ); 


} 


private SeparateSubTask sp = null; 
private TextField t = new TextField(10); 
private Button 


onoff 


new Button("Toggle"), 


start new Button("Start"); 

public void init() { 
add(t); 
start.addActionListener(new StartL()); 
add(start); 
onOff.addActionListener(new OnOffL()); 
add(onOff); 

} 

class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) 

if(sp == null) 


sp = new SeparateSubTask(); 


} 


class OnoffL implements ActionListener { 


public void actionPerformed(ActionEvent e) { 
if(sp != null) 


sp.runFlag = !sp.runFlag; // invertFlag(); 


} 


public static void main(String[] args) { 
Counter2i applet = new Counter2i(); 
Frame aFrame = new Frame("Counter2i"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 


aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(300, 200); 

applet.init(); 

applet.start(); 

aFrame.setVisible(true); 


} 
VS iP 


这 个 SeparateSubTask 名 字 不 会 与 前 例 中 的 SeparateSubTask 冲 突 一 一 即使 


它们 都 在 相同 的 目录 里 因为 它 己 作 为 一 个 内 部 类 隐藏 起 来 。 大 家 亦 
可 看 到 内 部 类 被 设 为 private〈 私 有 ) 属性， 这 意味 着 它 的 字段 和 方法 都 
可 获得 默认 的 访问 权限 〈run0 除 外 ， 它 必须 设 为 public， 因 为 它 在 基础 
类 中 是 公开 的 ) 。 除 Counter2i 之 外 ， 其 他 任何 方面 都 不 可 访问 private 内 
部 类 。 而 且 由 于 两 个 类 紧密 结合 在 一 起 ， 所 以 很 容易 放宽 它们 之 间 的 访 
问 限制 。 在 SeparateSubTask 中 ， 我 们 可 看 到 invertFlagO) 方 法 已 被 删 去 ， 
为 Counter2i 现 在 可 以 直接 访问 runFlag。 


此 外 ， 注 意 SeparateSubTask 的 构建 器 已 得 到 了 简化 它 现 在 唯一 的 用 
外 就 是 启动 线程 。Counter2i 对 象 的 句柄 仍 象 以 前 那样 得 以 捕获 ， 但 不 再 
是 通过 人 工 传递 和 引用 外 部 对 象 来 达到 这 一 目的 ， 此 时 的 内 部 类 机 制 可 
以 自动 照料 它 。 在 run0 中 ， 可 看 到 对 t 的 访问 是 直接 进行 的 ， 似 乎 它 是 
SeparateSubTask 的 一 个 字段 。 父 类 中 的 t 字 段 现 在 可 以 变 成 private， 因 为 
SeparateSubTask 能 在 未 获 任何 特殊 许可 的 前 提 下 自由 地 访问 它 而 且 
无 论 如 何 都 该 尽 可 能 地 把 字段 变 成 “私有 ”属性 ， 以 防 来 自 类 外 的 某 种 力 
量 不 慎 地 改变 它们 。 


无 论 在 什么 时 候 ， 只 要 注意 到 类 相互 之 间 络 合 得 比较 紧密 ， 就 可 考虑 利 
用 内 部 类 来 改善 代码 的 编写 与 维护 。 


14.1.3 用 主 类 合并 线程 


在 上 面 的 例子 中 ， 我 们 看 到 线程 类 (Thread) 与 程序 的 主 类 (Main) 是 
分 隔 开 的 。 这 样 做 非常 合理 ， 而 且 易 于 理解 。 然 而 ， 还 有 另 一 种 方式 也 
是 经 和 常 要 用 到 的 。 尺 管 它 不 十 分 明确 ， 但 一 般 都 要 更 简洁 一 些 ( 这 也 解 
释 了 它 为 什么 十 分 流行 ) 。 通 过 将 主 程序 类 变 成 一 个 线程 ， 这 种 形式 可 
将 主 程序 类 与 线程 类 合并 到 一 起 。 由 于 对 一 个 GUI 程序 来 说 ， 主 程序 类 
必须 从 Frame 或 Applet 继 承 ， 所 以 必须 用 一 个 接口 加 入 额外 的 功能 。 这 
个 接口 叫 作 Runnable， 其 中 包含 了 与 Thread 一 致 的 基本 方法 。 事 实 上 ， 
Thread 也 实现 了 Runnable， 它 只 指出 有 一 个 run0) 方 法 。 


对 合并 后 的 程序 / 线程 来 说 ， 它 的 用 法 不 是 十 分 明确 。 当 我 们 启动 程序 
时 ， 会 创建 一 个 Runnable (可 运行 的 ) 对 象 ， 但 不 会 自行 启动 线程 。 线 
程 的 启动 必须 明确 进行 。 下 面 这 个 程序 辐 我 们 演示 了 这 一 点 ， 它 再 现 了 
Counter2 的 功能 : 


















































//: Counter3.java 


// Using the Runnable interface to turn the 

// main class into a thread. 

import java.awt.*; 

import java.awt.event.*; 

import java.applet.*; 

public class Counter3 

extends Applet implements Runnable { 

private int count = 0; 
private boolean runFlag = true; 
private Thread selfThread = null; 
private Button 


onoff 


new Button("Toggle"), 


start new Button("Start"); 

private TextField t = new TextField(10); 

public void init() { 
add(t); 
start.addActionListener(new StartL()); 
add(start); 
onOff.addActionListener(new OnOffL()); 
add(onOff); 

} 


public void run() { 


while (true) { 


try { 


selfThread.sleep(100); 
} catch (InterruptedException e){} 
if(runFlag) 


t.setText(Integer.toString(count++) ); 


} 


class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(selfThread == null) { 
selfThread = new Thread(Counter3.this); 


selfThread.start(); 


} 


class OnoffL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


runFlag = !runFlag; 


} 


public static void main(String[] args) { 
Counter3 applet = new Counter3(); 


Frame aFrame = new Frame("Counter3"); 


aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(300, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
} ///:~ 


现在 run0 位 于 类 内 ， 但 它 在 init0 结 束 以 后 仍 处 在 “睡眠 "状态 。 若 按 下 启 
校 钮 ， 线 程 便 会 用 多 少 有 些 蚂 味 的 表达 方式 创建 着 线程 尚 不 丰 
> 


new Thread(Counter3.this); 


知 某 样 东 西 有 一 个 Runnable 接 口 ， 实 际 只 是 意味 着 它 有 一 个 run(0) 方 法 ， 

但 不 存在 与 之 相关 的 任何 特殊 东西 它 不 具有 任何 天 生 的 线程 处 理 能 
力 ， 这 与 那些 从 Thread 继 承 的 类 是 不 同 的 。 所 以 为 了 从 一 个 Runnable 对 
象 产 生 线 程 ， 必 须 单独 创建 一 个 线程 ， 并 为 其 传递 Runnable 对 象 ， 可 为 
其 使 用 一 个 特殊 的 构建 器 ， 并 令 其 采用 一 个 Runnable 作 为 自己 的 参数 使 
用 。 随 后 便 可 为 那个 线程 调用 start0， 如 下 所 示 : 


selfThread.start(); 























它 的 作用 是 执行 第 规 初 始 化 操作 ， 然 后 调用 run()。 


Runnable 接 口 最 大 的 一 个 优点 是 所 有 东西 部 从 属于 相同 的 类 。 夺 项 访问 

什么 东西 ， 只 需 简 单 地 访问 它 即 可 ， 不 需要 涉及 一 个 独立 的 对 象 。 但 为 

这 种 便利 也 是 要 付出 代价 的 一 一 只 可 为 那个 特定 的 对 象 运行 单独 一 个 线 

a 
Jos 


注意 Runnable 接 口 本 和 喘 并 不 是 造成 这 一 限制 的 非 魁 神童 。 它 古 由 于 
Runnable 与 我 们 的 主 类 合并 造成 的 ， 因 为 每 个 应 用 只 能 主 类 的 一 个 对 
象 。 











14.1.4 制作 多 个 线程 


现在 考虑 一 下 创建 多 个 不 同 的 线程 的 问题 。 我 们 不 可 用 前 面 的 例子 来 做 
到 这 一 点 ， 所 以 必须 倒退 回去 ， 利 用 从 Thread 继 承 的 多 个 独立 类 来 封 逆 
run()。 但 这 是 一 种 更 常规 的 方案 ， 而 且 更 易 理 解 ， 所 以 尽管 前 例 揭示 了 
我 们 经 党 都 能 看 到 的 编码 样式 ， 但 并 不 推荐 在 大 多 数 情况 下 都 那样 做 ， 
因为 它 只 是 稍微 复杂 一 些 ， 而 且 灵 活性 稍 低 一 些 。 


下 面 这 个 例子 用 计数 器 和 切换 按钮 再 现 了 前 面 的 编码 样式 。 但 这 一 次 ， 
一 个 特定 计数 占 的 所 有 信息 (按钮 和 文本 字段 ， 都 位 于 它 自 己 的 、 从 
Thread 继 承 的 对 象 内 。Ticker 中 的 所 有 字段 都 具有 private〈 私 有 ) 属 
性 ， 这 意味 着 Ticker 的 具体 实现 方案 可 根据 实际 情况 任意 修改 ， 其 中 包 
括 修改 用 于 获取 和 显示 信息 的 数据 组 件 的 数量 及 类 型 。 创 建 好 一 个 
Ticker 对 象 以 后 ， 构 建 器 便 请 求 一 个 AWT 容 器 (Container) 的 句柄 
Ticker 用 上 自己 的 可 视 组 件 填 充 那 个 容器 。 采 用 这 种 方式 ， 以 后 一 旦 改变 
了 可 视 组 件 ， 使 用 Ticker 的 代码 便 不 需要 另行 修改 一 道 。 


//: Counter4.java 




















// If you separate your thread from the main 
// class, you can have as many threads as you 
// want. 


import java.awt.*; 


import java.awt.event.*; 
import java.applet.*; 
class Ticker extends Thread { 
private Button b = new Button("Toggle"); 
private TextField t = new TextField(10); 
private int count = 0; 
private boolean runFlag = true; 
public Ticker(Container c) { 
b.addActionListener(new ToggleL()); 
Panel p = new Panel(); 
p.add(t); 
p.add(b); 
c.add(p); 
} 
class ToggleL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


runFlag = !runFlag; 


} 


public void run() { 
while (true) { 
if(runFlag) 


t.setText(Integer.toString(count++)); 


try { 


sleep(100); 


} catch (InterruptedException e){} 


} 


public class Counter4 extends Applet { 
private Button start = new Button("Start"); 
private boolean started = false; 
private Ticker[] s; 
private boolean isApplet = true; 
private int size; 
public void init() { 
// Get parameter "size" from Web page: 
if(isApplet ) 
size = 
Integer .parseInt(getParameter("size")); 
s = new Ticker[size]; 
for(int i = 0; i < s.length; i++) 
s[i] = new Ticker(this); 
start.addActionListener(new StartL()); 


add(start); 


class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(!started) { 
started = true; 
for(int i = 0; i < s.length; i++) 


s[i].start(); 


} 


public static void main(String[] args) { 
Counter4 applet = new Counter4(); 
// This isn't an applet, so set the flag and 
// produce the parameter values from args: 
applet.isApplet = false; 
applet.size = 
(args.length = 0 ? 5 : 
Integer.parseInt(args[0])); 
Frame aFrame = new Frame("Counter4"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(200, applet.size * 50); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
} ///:~ 


Ticker 不 仅 包 括 了 目 己 的 线程 处 理 机 制 ， 也 提供 了 控制 与 显示 线程 的 工 
ET E e 


在 Counter4 中 ， 有 一 个 名 为 s 的 Ticker 对 象 的 数组 。 为 获得 最 大 的 灵活 
性 ， 这 个 数组 的 长 度 是 用 程序 片 参数 接触 Web 页 而 初始 化 的 。 下 面 是 网 
本 
<applet code=Counter4 width=600 height=600> 

<param name=size value="20"> 

</applet> 

其 中 ，param，name 和 value 是 所 有 Web 页 都 适用 的 关键 字 。name 是 指 程 
序 中 对 参数 的 一 种 引用 称谓 ，value 可 以 是 任何 字 串 《并 不 仅仅 是 解析 成 
一 个 数字 的 东西 ) 。 

我 们 注意 到 对 数组 s 长 度 的 判断 是 在 init0 内 部 完成 的 ， 它 没有 作为 s 的 内 
骨 定 义 的 一 部 分 提供 。 换 言 之 ， 不 可 将 下 述 代码 作为 类 定义 的 一 部 分 使 
用 《应 该 位 于 任何 方法 的 外 部 ) : 


inst size = Integer.parseInt(getParameter("Size")); 


Ticker[] s = new Ticker|size] 


可 把 它 编译 出 来 ， 但 会 在 运行 期 得 到 一 个 空 指针 违例 。 但 知 将 
getParameter() 初 始 化 移入 init()， 则 可 正常 工作 。 程 序 片 框架 会 进行 必要 
的 启动 工作 ， 以 便 在 进入 init0 前 收集 好 一 些 参数 。 


此 外 ， 上 述 代码 被 同时 设置 成 一 个 程序 片 和 一 个 应 用 (程序) 。 在 它 是 
ee Size 参数 可 从 命令 行 里 提取 出 来 “否则 惑 提 供 一 个 
AARE) 。 


数组 的 长 度 建 好 以 后 ， 束 可 以 创建 新 的 Ticker 对 象 ， 作 为 Ticker 构 建 右 
的 一 部 分 ， 用 于 每 个 Ticker 的 按钮 和 文本 字段 就 会 加 入 程序 片 。 


按 下 Start 按 钮 后 ， 会 在 整个 Ticker 数 组 里 遍历 ， 并 为 每 个 Ticker 调 用 
start()。 记 住 ，start() 会 进行 必要 的 线程 初始 化 工作 ， 然 后 为 那个 线程 调 
用 run0)。 


TogglelL 监 视 器 只 是 简单 地 切换 Ticker 中 的 标记 ， 一 旦 对 应 线程 以 后 需要 
修改 这 个 标记 ， 它 会 作出 相应 的 反应 。 


这 个 例子 的 一 个 好 处 是 它 使 我 们 能 够 方便 地 创建 由 单独 子 任 务 构成 的 大 
型 集合 ， 并 以 监视 它们 的 行为 。 在 这 种 情况 下 ， 我 们 会 发 现 随 着 子 任务 
数量 的 增多 ， 机 器 显示 出 来 的 数字 可 能 会 出 现 更 大 的 分 鉴 ， 这 是 由 于 为 
线程 提供 服务 的 方式 造成 的 。 


亦 可 试 着 体验 一 下 sleep(100) 在 Ticker.run0 中 的 重要 作用 。 若 删除 
sleep()， 那 么 在 按 下 一 个 切换 按钮 前 ， 情 况 仍然 会 进展 展 好 。 按 下 按钮 
以 后 ， 那 个 特定 的 线程 就 会 出 现 一 个 失败 的 runFlag， 而 且 run0 会 深 深 地 
陷入 一 个 无 限 循环 很 难 在 多 任务 处 理 期 间 中 止 退 出 。 因 此 ， 程 序 对 
用 户 操作 的 反应 灵敏 度 会 大 幅度 降低 。 


14.1.5 Daemon 线 程 


“Daemon” 线 程 的 作用 是 在 程序 的 运行 期 间 于 后 台 提 供 一 种 “常规 ”服务 ， 
但 它 并 不 属于 程序 的 一 个 基本 部 分 。 因 此 ， 一 旦 所 有 非 Daemon 线 程 完 
成 ， 程 序 也 会 中 止 运 行 。 相 反 ， 假 符 有 任何 非 Daemon 线 程 仍 在 运行 
《比如 还 有 一 个 正在 运行 main0 的 线程 )》 ， 则 程序 的 运行 不 会 中 止 。 




















通过 调用 isDaemon()， 可 调查 一 个 线程 是 不 是 一 个 Daemon， 而 且 能 用 
setDaemon() 打 开 或 者 关闭 一 个 线程 的 Daemon 状 态 。 如 果 是 一 个 Daemon 
线程 ， 那 么 它 创 建 的 任何 线程 也 会 自动 具备 Daemon 属 性 。 


下 面 这 个 例子 演示 了 Daemon 线 程 的 用 法 : 


//: Daemons.java 


// Daemonic behavior 
import java.io.*; 
class Daemon extends Thread { 
private static final int SIZE = 10; 
private Thread[] t = new Thread[SIZE]; 
public Daemon() { 
setDaemon(true); 
start(); 
J 
public void run() { 
for(int i = 0; i < SIZE; i++) 
t[i] = new DaemonSpawn(i); 
for(int i = 0; i < SIZE; i++) 
System.out.printin( 
"tp" + i + "].isDaemon() = " 
+ t[1].isDaemon()); 
while(true) 


yield(); 


} 


class DaemonSpawn extends Thread { 

public DaemonSpawn(int i) { 

System.out.printin( 
"DaemonSpawn " + i + " started"); 

start(); 

} 

public void run() { 
while(true) 


yield(); 


J 


public class Daemons { 
public static void main(String[] args) { 
Thread d = new Daemon(); 
System.out.printin( 
"d.isDaemon() = " + d.isDaemon()); 
// Allow the daemon threads to finish 
// their startup processes: 
BufferedReader stdin = 
new BufferedReader ( 


new InputStreamReader(System.in)); 


System.out.println("Waiting for CR"); 
try { 
stdin.readLine(); 
} catch(IOException e) {} 
i, 


} ///:~ 


Daemon 线 程 可 将 自己 的 Daemon 标 记 设 置 成 * 真 >”， 然 后 产生 一 系列 其 他 
线程 ， 而 且 认 为 它们 也 具有 Daemon 属 性 。 随 后 ， 它 进入 一 个 无 限 循 
环 ， 在 其 中 调用 yield()， 放 弃 对 其 他 进程 的 控制 。 在 这 个 程序 早期 的 一 
个 版 本 中 ， 无 限 循环 会 使 int 计 数 器 增值 ， 但 会 使 整个 程序 都 好 象 陷入 停 
顿 状态 。 换 用 yieldO0 后 ， 却 可 使 程序 充满 “活力 ”， 不 会 使 人 产生 停 清 或 
反应 迟钝 的 感觉 。 





一 旦 main0 完 成 自己 的 工作 ， 便 没有 什么 能 阻止 程序 中 断 运 行 ， 因 为 这 
里 运行 的 只 有 Daemon 线 程 。 所 以 能 看 到 启动 所 有 Daemon 线 程 后 显示 出 
来 的 结果 ，System.in 也 进行 了 相应 的 设置 ， 使 程序 中 断 前 能 等 待 一 个 回 
车 。 如 果 不 进行 这 样 的 设置 ， 就 只 能 看 到 创建 Daemon 线 程 的 一 部 分 结 
果 《〈 试 试 将 readLine0 代 码 换 成 不 同 长 度 的 sleepO 调 用 ， 看 看 会 有 什么 表 
现 ) 。 


14.2 共享 有 限 的 资源 


可 将 单线 程 程序 想象 成 一 种 孤立 的 实体 ， 它 能 遍历 我 们 的 问题 空间 ， 而 
且 一 次 只 能 做 一 件 事情 。 由 于 只 有 一 个 实体 ， 所 以 永远 不 必 担心 会 有 两 
个 实体 同时 试图 使 用 相同 的 资源 ， 就 象 两 个 人 同时 都 想 停 到 一 个 车 位 ， 
同时 都 想 通 过 一 局 门 ， 甚 至 同时 发 话 。 


进入 多 线程 环境 后 ， 它 们 则 再 也 不 是 孤立 的 。 可 能 会 有 两 个 甚至 更 多 的 
线程 试图 同时 同一 个 有 限 的 资源 。 必 须 对 这 种 潜在 资源 冲突 进行 预防 ， 
否则 就 可 能 发 生 两 个 线程 同时 访问 一 个 银行 帐号 ， 打 印 到 同一 台 计 算 
机 ， 以 及 对 同一 个 值 进行 调整 等 等 。 


14.2.1 资源 访问 的 错误 方法 


现在 考虑 换 成 妨 一 种 方式 来 使 用 本 章 频 楷 见 到 的 计数 右 。 在 下 面 的 例子 
中 ， 每 个 线程 都 包含 了 两 个 计数 器 ， 它 们 在 run0 里 增值 以 及 显示 。 除 此 
以 外 ， 我 们 使 用 了 Watcher 类 的 为 一 个 线程 。 它 的 作用 是 监视 计数 器 ， 
检查 它们 是 否 保持 相等 。 这 表面 是 一 项 无 意义 的 行动 ， 因 为 如 果 碍 看 代 
码 ， 就 会 发 现 计数 器 肯定 是 相同 的 。 但 实际 情况 却 不 一 定 如 此 。 下 面 是 
程序 的 第 一 个 版 本 : 


//: Sharing1.java 

















// Problems with resource sharing while threading 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
class TwoCounter extends Thread { 
private boolean started = false; 
private TextField 


t1i = new TextField(5), 


t2 = new TextField(5); 
private Label 1 = 
new Label("counti == count2"); 
private int count1 = ©, count2 = 0; 
// Add the display components as a panel 
// to the given container: 
public TwoCounter(Container c) { 
Panel p = new Panel(); 
p.add(t1); 
p.add(t2); 
p.add(1); 
c.add(p); 
} 
public void start() { 
if(!started) { 
started = true; 


super.start(); 


} 


public void run() { 
while (true) { 
t1.setText(Integer.toString(counti++) ); 


t2.setText(Integer.toString(count2++) ); 


try { 


sleep(500); 


} catch (InterruptedException e){} 


} 


public void synchTest() { 
Sharing1.incrementAccess()/; 
if(counti != count2) 


1.setText("Unsynched") ; 


J 


class Watcher extends Thread { 
private Sharing1 p; 
public Watcher(Sharingi p) { 
this.p = p; 
start(); 
} 
public void run() { 
while(true) { 
for(int i = 0; i < p.s.length; i++) 
p.s[i].synchTest(); 
try { 


sleep(500); 


} catch (InterruptedException e){} 


J 


public class Sharingi extends Applet { 
TwoCounter[] s; 
private static int accessCount = 0; 
private static TextField aCount = 
new TextField("0", 10); 
public static void incrementAccess() { 
accessCount++; 
aCount.setText(Integer.toString(accessCount ) ); 
} 
private Button 
start = new Button("Start"), 
observer = new Button("Observe"); 
private boolean isApplet = true; 
private int numCounters = 0; 
private int numObservers = 0; 
public void init() { 
if(isApplet) { 
numCounters = 


Integer.parseInt(getParameter("size")); 


numObservers = 
Integer .parseInt( 
getParameter("observers")); 
} 
s = new TwoCounter[numCounters]; 
for(int i = 0; i < s.length; i++) 
s[i] = new TwoCounter(this) ; 
Panel p = new Panel(); 
start.addActionListener(new StartL()); 
p.add(start); 
observer .addActionListener(new ObserverL()); 
p.add(observer); 
p.add(new Label("Access Count")); 
p.add(aCount); 
add(p); 
} 
class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
for(int i = 0; i < s.length; i++) 


s[i].start(); 


} 


Class ObserverL implements ActionListener { 


public void actionPerformed(ActionEvent e) { 
for(int i = 0; i < numObservers; i++) 


new Watcher(Sharing1.this); 


} 


public static void main(String[] args) { 
Sharing1 applet = new Sharing1(); 
// This isn't an applet, so set the flag and 
// produce the parameter values from args: 
applet.isApplet = false; 
applet.numCounters = 
(args.length = 0 ? 5 : 
Integer.parseInt(args[0])); 
applet.numObservers = 
(args.length <2? 5: 
Integer.parseInt(args[1])); 
Frame aFrame = new Frame("Sharingi"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 


System.exit(0); 


}); 


aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(350, applet.numCounters *100); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
A 








和 往常 一 样 ， 每 个 计数 器 都 包含 了 自己 的 显示 组 件 ， 两 个 文本 字段 以 及 
一 个 标签 。 根 据 它们 的 初始 值 ， 可 知道 计数 是 相同 的 。 这 些 组 件 在 
TwoCounter 构 建 器 加 入 Container。 由 于 这 个 线程 是 通过 用 户 的 一 个 “ 按 
下 按钮 ”操作 启动 的 ， 所 以 start() 可 能 被 多 次 调用 。 但 对 一 个 线程 来 说 ， 
对 Thread.start() 的 多 次 调用 是 非法 的 (会 产生 违例 ) 。 在 started 标 记 和 过 
载 的 start() 方 法 中 ， 大 家 可 看 到 针对 这 一 情况 采取 的 防范 措施 。 


在 run0 中 ，count1 和 count2 的 增值 与 显示 方式 表面 上 似乎 能 保持 它们 完 
全 一 致 。 随 后 会 调用 sleep0; 奋 没 有 这 个 调用 ， 程 序 便 会 出 错 ， 因 为 那 
会 造成 CPU 难于 交换 任务 。 


synchTest(0 方 法 采取 的 似乎 是 没有 意义 的 行动 ， 它 检查 count1 是 否 等 于 
count2; 如 果 不 等 ， 就 把 标签 设 为 “Unsynched”( 不 同步 ) 。 但 是 首先 ， 
它 调 用 的 是 类 Sharing1 的 一 个 静态 成 员 ， 以 便 增 值 和 显示 一 个 访问 计数 
器 ， 指 出 这 种 检查 已 成 功 进行 了 多 少 次 《这样 做 的 理由 会 在 本 例 的 其 他 
版 本 中 变 得 非常 明显 ) 。 


Watcher 类 是 一 个 线程 ， 它 的 作用 是 为 处 于 活动 状态 的 所 有 TwoCounter 
对 象 都 调用 synchTest()。 其 间 ， 它 会 对 Sharing1 对 象 中 容纳 的 数组 进行 
遍历 。 可 将 Watcher 想 象 成 它 掠 过 TwoCounter 对 象 的 肩膀 不 断 地 “ 丛 

看 ” y 


Sharing1 包 含 了 TwoCounter 对 象 的 一 个 数组 ， 它 通过 initO 进 行 初始 化 ， 
并 在 我 们 按 下 “start” 按 钮 后 作为 线程 启动 。 以 后 奇 按 下 “Observe”( 观 
Be) 按钮 ， 就 会 创建 一 个 或 者 多 个 观察 器 ， 并 对 晓 不 设防 的 TWoCounter 











进行 调查 。 


注意 为 了 证 它 作 为 一 个 程序 片 在 浏览 器 中 运行 ，Web 页 需要 包含 下 面 这 
JLAT: 


<applet code=Sharingi width=650 height=500> 





<param name=size value="20"> 
<param name=observers value="1"> 


</applet> 

















可 自行 改变 宽度 、 高 度 以 及 参数 ， 根 据 上 自己 的 意愿 进行 试验 。 若 改变 了 
size 和 observers， 程 序 的 行为 也 会 发 生变 化 。 我 们 也 注意 到 ， 通 过 从 命 

参数 〈 或 者 使 用 默认 值 ) ， 它 被 设计 成 作为 一 个 独立 的 应 用 程 
运行 


下 面 才 是 最 让 人 “不 可 思议 ”的 。 在 TwocCounter.run0 中 ， 无 限 循环 只 是 不 
bap Sty HEL IS ARQ AY 4 : 


t1.setText(Integer.toString(count1++)); 
t2.setText(Integer.toString(count2++)); 


4 和 “睡眠 ”一 样 ， 不 过 在 这 里 并 不 重要 ) 。 但 在 程序 运行 的 时 候 ， 你 会 
发 现 count1 和 count2 被 “观察 ”( 用 Watcher 观 察 ) 的 次 数 是 不 相等 的 ! 这 
是 由 线程 的 本 质 造 成 zA =) 。 所 以 在 上 
述 两 行 的 执行 时 刻 之 间 ， 有 时 会 出 现 执行 暂停 现象 。 同 时 ，Watcher 线 
并 正好 在 这 个 时 候 进 行 比较 ， 造 成 计数 器 出 现 不 
日 等 的 情况 。 


本 例 揭示 了 使 用 线程 时 一 个 非常 基本 的 问题 。 我 们 跟 无 从 知道 一 个 线程 
什么 时 候 运 行 。 想 象 自己 坐 在 一 张 虹 子 前 面 ， 果 上 放 有 一 把 久子， 准备 
又 起 自己 的 最 后 一 块 食物 。 当 叉子 要 磁 到 食物 时 ， 食 物 却 突然 消失 了 
(因为 这 个 线程 已 被 挂 起 ， 同 时 男 一 个 线程 进来 “ 偷 ”* 走 了 食物 ) 。 这 便 
古 我 们 要 解决 的 问题 。 








有 的 时 候 ， 我 们 并 不 介意 一 个 资源 在 尝试 使 用 它 的 时 候 是 否 正 被 访问 
《食物 在 另 一 些 盘子 里 ) 。 但 为 了 让 多 线程 机 制 能 够 正常 运转 ， 需 要 和 采 
取 一 些 措 施 来 防止 两 个 线程 访问 相同 的 资源 一 一 全 少 在 关键 的 时 期 。 


为 防止 出 现 这 样 的 冲突 ， 只 雷 在 线程 使 用 一 个 资源 时 为 其 加 锁 即 可 。 访 
问 资源 的 第 一 个 线程 会 其 加 上 锁 以 后 ， 其 他 线程 便 不 能 再 使 用 那个 资 
源 ， 除 非 被 解锁 。 如 果 车 子 的 前 座 是 有 限 的 资源 ， 高 喊 “ 这 是 我 的 ! ”的 
孩子 会 主张 把 它 锁 起 来 。 


14.2.2 Java 如 何 共 享 资源 


对 一 种 特殊 的 资源 一 一 对 象 中 的 内 存 一 一 Java 提 供 了 内 建 的 机 制 来 防止 
它们 的 冲突 。 由 于 我 们 通常 将 数据 元 素 设 为 从 属于 private ALA) 类 ， 
然后 只 通过 方法 访问 那些 内 存 ， 所 以 只 需 将 一 个 特定 的 方法 设 为 
synchronized (同步 的 ) ， 便 可 有 效 地 防止 冲突 。 在 任何 时 刻 ， 只 可 有 
一 个 线程 调用 特定 对 象 的 一 个 synchronized 方 法 (尽管 那个 线程 可 以 调 
用 多 个 对 象 的 同步 方法 ) 。 下 面 列 出 简单 的 synchronized 方 法 : 


synchronized void f() { /* ... */ } 











synchronized void g() { /* ... */ } 


PERM RAMS SE (也 叫 作 “监视 器 *) ， 它 自动 成 为 对 象 的 一 部 
分 (不 必 为 此 写 任 何 特殊 的 代码 ) 。 调 用 任何 synchronized 方 法 时 ， 对 
象 就 会 被 锁定 ， 不 可 再 调用 那个 对 象 的 其 他 任何 synchronized 方 法 ， 除 
非 第 一 个 方法 完成 了 自己 的 工作 ， 并 解除 锁定 。 在 上 面 的 例子 中 ， 如 果 
为 一 个 对 象 调用 f()， 便 不 能 再 为 同样 的 对 象 调用 g()， 除 非 f0 完 成 并 解除 
锁定 。 因 此 ， 一 个 特定 对 象 的 所 有 synchronized 方 法 都 共享 着 一 把 锁 ， 
ee aa 内 存 同 时 进行 写 操 作 〈( 比 如 同时 有 多 
上 线程 ) 。 


每 个 类 也 有 自己 的 一 把 锁 〈 作 为 类 的 Class 对 象 的 一 部 分 》 ， 所 以 
synchronized static 方 法 可 在 一 个 类 的 范围 内 被 相互 间 锁 定 起 来 ， 防 止 与 
static 数 据 的 接触 。 


注意 如 果 想 保护 其 他 某 些 资源 不 被 多 个 线程 同时 访问 ， 可 以 强制 通过 
synchronized 方 访问 那些 资源 。 














1. 计数 器 的 同步 

装备 了 这 个 新 关键 字 后 ， 我 们 能 够 采取 的 方案 就 更 灵活 了 : 可 以 只 为 
TwoCounter 中 的 方法 简单 地 使 用 synchronized 关 键 字 。 下 面 这 个 例子 是 
对 前 例 的 改版 ， 其 中 加 入 了 新 的 关键 字 : 


//: Sharing2.java 








// Using the synchronized keyword to prevent 
// multiple access to a particular resource. 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
class TwoCounter2 extends Thread { 
private boolean started = false; 
private TextField 
ti = new TextField(5), 
t2 = new TextField(5); 
private Label 1 = 
new Label("counti == count2"); 
private int counti = 0, count2 = 0; 
public TwoCounter2(Container c) { 
Panel p = new Panel(); 
p.add(t1); 
p.add(t2); 


p.add(1); 


c.add(p); 
ae 
public void start() { 
if(!started) { 
started = true; 


super.start(); 


j 


public synchronized void run() { 
while (true) { 
t1.setText(Integer.toString(countí++)); 
t2.setText(Integer.toString(count2++) ); 
try { 
sleep(500); 


} catch (InterruptedException e){} 


} 


public synchronized void synchTest() { 
Sharing2.incrementAccess(); 
if(count1 != count2) 


l.setText("Unsynched"); 


class Watcher2 extends Thread { 
private Sharing2 p; 
public Watcher2(Sharing2 p) { 
this.p = p; 
start(); 
} 
public void run() { 
while(true) { 
for(int i = 0; i < p.s.length; i++) 
p.s[1i].synchTest(); 
try { 
sleep(500); 


} catch (InterruptedException e){} 


} 
public class Sharing2 extends Applet { 
TwoCounter2[] s; 
private static int accessCount = 0; 
private static TextField aCount = 
new TextField("0", 10); 
public static void incrementAccess() { 


accessCount++,; 


aCount.setText(Integer.toString(accessCount) ); 
} 
private Button 
start = new Button("Start"), 
observer = new Button("Observe"); 
private boolean isApplet = true; 
private int numCounters = 0; 
private int numObservers = 0; 
public void init() { 
if(isApplet) { 
numCounters = 
Integer .parseInt(getParameter("size")); 
numObservers = 
Integer .parseInt( 
getParameter ("observers") ); 
} 
s = new TwoCounter2[numCounters]; 
for(int i = 0; i < s.length; i++) 
s[i] = new TwoCounter2(this); 
Panel p = new Panel(); 
start.addActionListener(new StartL()); 
p.add(start); 


observer .addActionListener(new ObserverL()); 


p.add(observer ) ; 
p.add(new Label("Access Count")); 
p.add(aCount); 
add(p); 

} 

class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 

for(int i = 0; i < s.length; i++) 


s[i].start(); 


} 


Class ObserverL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
for(int i = 0; i < numObservers; i++) 


new Watcher2(Sharing2.this); 


} 


public static void main(String[] args) { 
Sharing2 applet = new Sharing2(); 
// This isn't an applet, so set the flag and 
// produce the parameter values from args: 
applet.isApplet = false; 


applet.numCounters = 


(args.length = 0 ? 5 : 
Integer.parseInt(args[0])); 
applet.numObservers = 
(args.length <2? 5: 
Integer.parseInt(args[1])); 
Frame aFrame = new Frame("Sharing2"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
System.exit(0); 
} 
}); 


aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(350, applet.numCounters *100); 
applet.init(); 

applet.start(); 

aFrame.setVisible(true); 


} 
} ///:~ 


我 们 注意 到 无 论 run0 还 是 synchTestO 都 是 “同步 的 ”。 如 果 只 同步 其 中 的 
一 个 方法 ， 那 么 男 一 个 就 可 以 自由 忽视 对 象 的 锁定 ， 并 可 无 碍 地 调用 。 
所 以 必须 记 住 一 个 重要 的 规则 : 对 于 访问 某 个 关键 共享 资源 的 所 有 方 
法 ， 都 必须 把 它们 设 为 synchronized， 人 否则 就 不 能 正常 地 工作 。 





现在 又 遇 到 了 一 个 新 间 题 。Watcher2 永 远 都 不 能 看 到 正在 进行 的 事情 ， 
因为 整个 run0) 方 法 已 设 为 “同步 ”。 而 且 由 于 肯定 要 为 每 个 对 象 运 行 
run()， 所 以 锁 永远 不 能 打开 ， 而 synchTest() 永 远 不 会 得 到 调用 。 之 所 以 
能 看 到 这 一 结果 ， 是 因为 accessCount 根 本 没有 变化 。 








为 解决 这 个 问题 ， 我 们 能 采取 的 一 个 办 法 是 只 将 run0 中 的 一 部 分 代码 隔 
离 出 来 。 想 用 这 个 办 法 隔离 出 来 的 那 部 分 代码 叫 作 “ 关 键 区 域 "， 而 且 要 
用 不 同 的 方式 来 使 用 synchronized 关 键 字 ， 以 设置 一 个 关键 区 域 。Java 通 
过 “同步 块 ” 提 供 对 关键 区 域 的 支持 ; 这 一 次 ， 我 们 用 synchronized 关 键 
字 指 出 对 象 的 锁 用 于 对 其 中 封闭 的 代码 进行 同步 。 如 下 所 示 : 





synchronized(syncObject) { 


// This code can be accessed by only 
// one thread at a time, assuming all 


// threads respect syncObject's lock 


} 


在 能 进入 同步 块 之 前 ， 必 须 在 synchObject 上 取得 锁 。 如 果 已 有 其 他 线程 
取得 了 这 把 锁 ， 块 便 不 能 进入 ， 必 须 等 候 那 把 锁 被 释放 。 


可 从 整个 run0 中 删除 synchronized 关 键 字 ， 换 成 用 一 个 同步 块 包围 两 个 
关键 行 ， 从 而 完成 对 Sharing2 例 子 的 修改 。 但 什么 对 象 应 作为 锁 来 使 用 
We? 那个 对 象 已 由 synchTest() 标 记 出 来 了 一 一 也 就 是 当前 对 象 Cthis) ! 
所 以 修改 过 的 mn0 方 法 象 下 面 这 个 样子 : 


public void run() { 


while (true) { 
synchronized(this) { 
t1.setText(Integer.toString(counti++) ); 


t2.setText(Integer.toString(count2++) ); 


} 
try { 
sleep(500); 


} catch (InterruptedException e){} 


这 是 必须 对 Sharing2.java 作 出 的 唯一 修改 ， 我 们 会 看 到 尽管 两 个 计数 器 
永远 不 会 脱离 同步 〈 取 决 于 允许 Watcher 什 么 时 候 检 查 它 们 ) ， 但 在 
run0 执 行 期 间 ， 仍 然 向 Watcher 提 供 了 足够 的 访问 权限 。 


当然 ， 所 有 同步 都 取决 于 程序 员 古 否 勤 奋 : 要 访问 共 孚 资源 的 每 一 部 分 
代码 都 必须 封装 到 一 个 适当 的 同步 块 里 。 


2. 同步 的 效率 


由 于 要 为 同样 的 数据 编号 两 个 方法 ， 所 以 无 论 如 何 都 不 会 给 人 留 下 效率 
很 高 的 印象 。 看 来 似乎 更 好 的 一 种 做 法 是 将 所 有 方法 都 设 为 自动 同步 ， 
并 完全 消除 synchronized 关 键 字 (当然 ， 含 有 synchronized run() 的 例子 显 
示 出 这 样 做 是 很 不 通 的 ) 。 但 它 也 揭示 出 获取 一 把 锁 并 非 一 种 “廉价 ” 方 
案 一 一 为 一 次 方法 调用 付出 的 代价 《进入 和 退出 方法 ， 不 执行 方法 主 
体 ) 至 少 要 累加 到 四 倍 ， 而 且 根 据 我 们 的 具体 现 方案 ， 这 一 代价 还 有 可 
能 变 得 更 高 。 所 以 假如 已 知 一 个 方法 不 会 造成 冲突 ， 最 明智 的 做 法 便 是 
撤消 其 中 的 synchronized 关 键 字 。 


14.2.3 回顾 Java Beans 


我 们 现在 已 理解 了 同步 ， 接 着 可 换 从 为 一 个 角 展 来 考察 Java Beans。 无 
论 什么 时 候 创 建 了 一 个 Bean， 就 必须 假定 它 要 在 一 个 多 线程 的 环境 中 运 
行 。 这 意味 着 : 


(1) ”只 要 可 行 ，Bean 的 所 有 公共 方法 都 应 同步 。 当 然 ， 这 也 市 来 了 “ 同 
步 " 在 运行 期 间 的 开销 。 夺 特别 在 意 这 个 问题 ， 在 关键 区 域 中 不 会 造成 
问题 的 方法 就 可 保留 为 “不 同步 "， 但 注意 这 通常 部 不 是 十 分 容易 判断 。 








有 资格 的 方法 倾 同 于 规模 很 小 (如 下 例 的 getCirdleSize()) 以 及 /或 

者 “微小 >。 也 就 是 说 ， 这 个 方法 调用 在 如 此 少 的 代码 片 里 执行 ， 以 至 于 
在 执行 期 间 对 象 不 能 改变 。 如 果 将 这 种 方法 设 为 “不 同步 ”， 可 能 对 程序 
的 执行 速度 不 会 有 明显 的 影响 。 可 能 也 将 一 个 Bean 的 所 有 public 方 法 都 
设 为 synchronized， 并 只 有 在 保证 特别 必要 、 而 且 会 造成 一 个 差异 的 情 
况 下 ， 才 将 synchronized 关 键 字 删 去 。 


D 如果 将 一 个 多 造型 事件 送 给 一 系列 对 那个 事件 感 兴趣 的 “听众 ”， 必 
须 假 在 列表 中 移动 的 时 候 可 以 添加 或 者 删除 。 


第 一 点 很 容易 处 理 ， 但 第 二 点 需要 考虑 更 多 的 东西 。 让 我 们 以 前 一 章 提 
供 的 BangBean.java 为 例 。 在 那个 例子 中 ， 我 们 忽略 了 synchronized 关 键 
字 【〔 那 时 还 没有 引入 呢 〉， 并 将 造型 设 为 单 造型 ， 从 而 回避 了 多 线程 的 
问题 。 在 下 面 这 个 修改 过 的 版 本 中 ， 我 们 使 其 能 在 多 线程 环境 中 工作 ， 
并 为 事件 采用 了 多 造型 技术 : 


//: BangBean2.java 




















// You should write your Beans this way so they 
// can run in a multithreaded environment. 
import java.awt.*; 
import java.awt.event.*; 
import java.util.*; 
import java.io.*; 
public class BangBean2 extends Canvas 
implements Serializable { 

private int xm, ym; 

private int cSize = 20; // Circle size 
private String text = "Bang!"; 


private int fontSize = 48; 


private Color tColor = Color.red; 
private Vector actionListeners = new Vector(); 
public BangBean2() { 
addMouseListener(new ML()); 
addMouseMotionListener(new MM()); 
} 
public synchronized int getCircleSize() { 
return cSize; 
} 
public synchronized void 
setCircleSize(int newSize) { 
cSize = newSize; 
} 
public synchronized String getBangText() { 
return text; 
} 
public synchronized void 
setBangText(String newText) { 
text = newText; 
} 
public synchronized int getFontSize() { 


return fontSize; 


public synchronized void 
setFontSize(int newSize) { 
fontSize = newSize; 
} 
public synchronized Color getTextColor() { 
return tColor; 
} 
public synchronized void 
setTextColor(Color newColor) { 
tColor = newColor; 
} 
public void paint(Graphics g) { 
g.setColor(Color.black); 
g.drawOval(xm - cSize/2, ym - cSize/2, 
cSize, cSize); 
} 
// This is a multicast listener, which is 
// more typically used than the unicast 
// approach taken in BangBean. java: 
public synchronized void addActionListener ( 
ActionListener 1) { 


actionListeners.addElement(1); 


public synchronized void removeActionListener ( 
ActionListener 1) { 
actionListeners.removeElement(1); 
} 
// Notice this isn't synchronized: 
public void notifyListeners() { 
ActionEvent a = 
new ActionEvent(BangBean2.this, 
ActionEvent.ACTION_PERFORMED, null); 
Vector lv = null; 
// Make a copy of the vector in case someone 
// adds a listener while we're 
// calling listeners: 
synchronized(this) { 
lv = (Vector )actionListeners.clone(); 
} 
// Call all the listener methods: 
for(int i = 0; i < lv.size(); i++) { 
ActionListener al = 
(ActionListener )lv.elementAt(i); 


al.actionPerformed(a); 


class ML extends MouseAdapter { 
public void mousePressed(MouseEvent e) { 
Graphics g = getGraphics(); 
g.setColor(tColor); 
g.setFont ( 
new Font ( 
"TimesRoman", Font.BOLD, fontSize)); 
int width = 
g.getFontMetrics().stringwidth(text); 
g.drawString(text, 
(getSize().width - width) /2, 
getSize().height/2); 
g.dispose(); 


notifyListeners(); 


} 


class MM extends MouseMotionAdapter { 
public void mouseMoved(MouseEvent e) { 
xm = e.getX(); 


e.getyY(); 


ym 


repaint(); 


// Testing the BangBean2: 
public static void main(String[] args) { 

BangBean2 bb = new BangBean2(); 

bb.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e){ 

System.out.printin("ActionEvent" + e); 

} 

}); 


bb.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e){ 
System.out.println("BangBean2 action"); 
} 
}); 


bb.addActionListener(new ActionListener() { 

public void actionPerformed(ActionEvent e){ 
System.out.println("More action"); 

} 

}); 

Frame aFrame = new Frame("BangBean2 Test"); 

aFrame.addwindowListener(new WindowAdapter ( ) { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


}); 
aFrame.add(bb, BorderLayout .CENTER); 
aFrame.setSize(300, 300); 


aFrame.setVisible(true); 


} 
} ///:~ 


很 容易 就 可 以 为 方法 添加 synchronized。 但 注意 在 addActionListener() 和 
removeActionListenerO0 中 ， 现 在 添加 了 ActionListener， 并 从 一 个 Vector 
中 移 去 ， 所 以 能 够 根据 自己 愿望 使 用 任意 多 个 。 


我 们 注意 到 ，notifyListeners(0) 方 法 并 未 设 为 “同步 ”。 可 从 多 个 线程 中 发 
出 对 这 个 方法 的 调用 。 另 外 ， 在 对 notifyListenersO) 调 用 的 中 途 ， 也 可 能 
发 出 对 addActionListener() 和 removeActionListener() 的 调用 。 这 显然 会 造 
成 问题 ， 因 为 它 否 定 了 Vector actionListeners。 为 绥 解 这 个 问题 ， 我 们 在 
一 个 synchronized 从 名 中 “克隆 ”了 Vector， 并 对 克隆 进行 了 否定。 这 样 便 
可 在 不 影响 notifyListenersO 的 前 提 下 ， 对 Vector 进行 操纵 。 


paint( 方 法 也 没有 设 为 “同步 ”。 与 单纯 地 添加 目 己 的 方法 相 比 ， 诀 定 是 
售 对 过 载 的 方法 进行 同步 要 困难 得 多 。 在 这 个 例子 中 ， 无 论 paintO 征 
否 “ 同 步 "， 它 似乎 都 能 正常 地 工作 。 但 必须 考虑 的 问题 包括 : 


(1) ”方法 会 在 对 象 内 部 修改 “关键 "变量 的 状态 吗 ? 为 判断 一 个 变量 是 
侍 “ 关 键 ”"”， 必 须知 道 它 是 否 会 被 程序 中 的 其 他 线程 读 取 或 设置 〈 束 目 前 

的 情况 看 ， 读 取 或 设置 几乎 肯定 是 通过 “同步 ”方法 进行 的 ， 所 以 可 以 只 

对 它们 进行 检查 ) 。 对 paintO 的 情况 来 说 ， 不 会 发 生 任何 修改 。 


(2) 方法 要 以 这 些 “ 关 键 " 变 量 的 状态 为 基础 吗 ? 如 果 一 个 “同步 ”方法 修改 
了 一 个 变量 ， 而 我 们 的 方法 要 用 到 这 个 变量 ， 那 么 一 般 都 愿意 把 目 己 的 
方法 也 设 为 “同步 *。 基 于 这 一 前 提 ， 大 家 可 观察 到 cSize 由 “同步 方法 进 
行 了 修改 ， 所 以 paint0 应 当 是 “同步 ”的 。 但 在 这 里 ， 我 们 可 以 问 :“ 假 如 
cSize 在 paint() 执 行 期 间 发 生 了 变化 ， 会 友 生 的 最 糟 料 的 事情 是 什么 

Ne? ?如果 发 现 情况 不 算 太 坏 ， 而 且 仅 仅 是 暂时 的 效果 ， 那 么 最 好 保持 

paint() 的 “不 同步 状态， 以 避免 同步 方法 调用 市 来 的 额外 开销 。 























(3) 要 留意 的 第 三 条 线索 是 paint0 基 础 类 版 本 是 否 “ 同 步 ”， 在 这 里 它 不 是 
同步 的 。 这 并 不 是 一 个 非常 严格 的 参数 ， Me 线索” 比如 在 目 
前 的 情况 下 ， 通过 同步 方法 《好 cSize) 改变 的 一 个 字段 已 合成 到 paintO 
公式 里 ， 而 且 可 能 已 改变 了 情况 。 但 请 注意 ，synchronized 不 能 继承 
一 一 也 就 是 说 ， 假 如 一 个 方法 在 基础 类 中 是 “同步 ”的 ， 那 么 在 衍生 类 过 
载 版 本 中 ， 它 不 会 自动 进入 “同步 ?状态 。 


TestBangBean2 中 的 测试 代码 已 在 前 一 章 的 基础 上 进行 了 修改 ， 己 在 其 
中 加 入 了 和 额外 的 “ 旷 众 ”?， 从 而 演示 了 BangBean2 的 多 造型 能 











14.3 堵塞 
一 个 线程 可 以 有 四 种 状态 : 
(1) # (New) : 线程 对 象 已 经 创建 ， 但 尚未 启动 ， 所 以 不 可 运行 。 


(2) 可 运行 (Runnable) : 意味 着 一 旦 时 间 分 片 机 制 有 空 闪 的 CPU 周 期 提 
供给 一 个 线程 ， 那 个 线程 便 可 立即 开始 运行 。 因此， 线程 可 能 在 、 也 可 
能 不 在 运行 当中 ， 但 一 旦 条 件 许可 ， 没 有 什么 能 阻止 它 的 运行 一 一 它 既 
A Ht, TAR BE DASE”. 


(3) JÉ (Dead) : 从 自己 的 run0 方 法 中 返回 后 ， 一 个 线程 便 已 “ 死 ” 掉 。 

亦 可 调用 stop0 令 其 死 掉 ， 但 会 产生 一 个 违例 一 一 属于 Error 的 一 个 子 类 

(也 就 是 说 ， 我 们 通常 不 捕获 它 ) 。 记 住 一 个 违例 的 “ 搓 ” 出 应 当 是 一 个 

特殊 事件 ， 而 不 是 正常 程序 运行 的 一 部 分 。 所 以 不 建议 你 使 用 stop() 

(在 Java 1.2 则 是 坚决 反对 ) 。 男 外 还 有 一 个 destroy() 方 法 ( 它 永 远 不 会 

a 应 该 尽 可 能 地 避免 调用 它 ， 因 为 它 非常 武断 ， 根 本 不 会 解除 对 
锁定 。 


(4) HÆ (Blocked) : 线程 可 以 运行 ， 但 有 某 种 东西 阻碍 了 它 。 知 线程 
处 于 堵塞 状态 ， 调 度 机 制 可 以 简单 地 跳 过 它 ， 不 给 它 分 配 任何 CPU 时 
间 。 除 非 线程 再 次 进入 “可 运行 ?状态 ， 否 则 不 会 采取 任何 操作 。 


14.3.1 为 何 会 堵塞 


堵 豆 状态 是 前 述 四 种 状态 中 最 有 趣 的 ， 值 得 我 们 作 进 一 步 的 探讨 。 线 程 
被 堵塞 可 能 是 由 下 述 五 方面 的 原因 造成 的 : 


(1) 调用 sleep( 坚 秒 数 )， 使 线程 进入 “睡眠 ?状态 。 在 规定 的 时 间 内 ， 这 个 
线程 是 不 会 运行 的 。 


(2) 用 suspend0O 和 暂停 了 线程 的 执行 。 除 非 线 程 收 到 resume0O 消 息 ， 人 否则 不 
会 返回 “可 运行 ”状态 。 


(3)” 用 wait0 和 暂停 了 线程 的 执行 。 除 非 线 程 收 到 nofifyO 或 者 notifyAlO 消 
息 ， 否 则 不 会 变 成 “可 运行 "(是 的 ， 这 看 起 来 同 原因 2 非常 相 象 ， 但 有 
一 个 明显 的 区 别 是 我 们 马上 要 揭示 的 ) 。 





























(4) 线程 正在 等 候 一 些 IO 输入 输出 ) 操作 完成 。 


(5) ”线程 试图 调用 男 一 个 对 象 的 “同步 方法 ， 但 那个 对 象 处 于 锁定 状 
态 ， 和 暂时 无 法 使 用 。 


亦 可 调用 yield() 〈Thread 类 的 一 个 方法 ) 自动 放弃 CPU， 以 便 其 他 线程 
能 够 运行 。 然 而 ， 假 如 调度 机 制 觉 得 我 们 的 线程 已 拥有 足够 的 时 间 ， 并 
跳 转 到 另 一 个 线程 ， 就 会 发 生 同 样 的 事情 。 也 就 是 说 ， 没 有 什么 能 防止 
调度 机 制 重新 启动 我 们 的 线程 。 线 程 被 堵塞 后 ， 便 有 一 些 原因 造成 它 不 
能 继续 运行 。 


下 面 这 个 例子 展示 了 进入 堵塞 状态 的 全 部 五 种 途径 。 它 们 全 都 存在 于 名 
为 Blocking.java 的 一 个 文件 中 ， 但 在 这 儿 采 用 散落 的 上 请 断 进行 解释 〈 大 
家 可 注意 到 片断 前 后 的 “Continued” 以 及 “Continuing” 标 志 。 利 用 第 17 章 
介绍 的 工具 ， 可 将 这 些 片 断 连 结 到 一 起 ) 。 肖 先 让 我 们 看 看 基本 的 框 
BR: 


//: Blocking.java 


// Demonstrates the various ways a thread 
// can be blocked. 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
import java.io.*; 
//////////// The basic framework /////////// 
Class Blockable extends Thread { 
private Peeker peeker; 
protected TextField state = new TextField(40); 


protected int i; 


public Blockable(Container c) { 
c.add(state); 
peeker = new Peeker (this, c); 
} 
public synchronized int read() { return i; } 
protected synchronized void update() { 
state.setText(getClass().getName() 
+ " state: i =" + i); 
} 
public void stopPeeker() { 
// peeker.stop(); Deprecated in Java 1.2 


peeker.terminate(); // The preferred approach 


} 


class Peeker extends Thread { 

private Blockable b; 

private int session; 

private TextField status = new TextField(40); 

private boolean stop = false; 

public Peeker(Blockable b, Container c) { 
c.add(status); 
this.b = b; 


start(); 


} 
public void terminate() { stop = true; } 
public void run() { 
while (!stop) { 
status.setText(b.getClass().getName() 
+ " Peeker " + (++session) 
+ "; value = " + b.read()); 
try { 
sleep(100); 


} catch (InterruptedException e){} 


} 


} ///:Continued 





Blockable 类 打算 成 为 本 例 所 有 类 的 一 个 基础 类 。 一 个 Blockable 对 象 包含 
了 一 个 名 为 state 的 TextField (文本 字段 ，， 用 于 显示 出 对 象 有 关 的 信 
息 。 用 于 显示 这 些 信息 的 方法 叫 作 update()。 我 们 发 现 它 用 
getClass.getName() 来 产生 类 名 ， 而 不 是 仅仅 把 它 打 印 出 来 ， 这 是 由 于 
update(0 不 知道 自己 为 其 调用 的 那个 类 的 准确 名 字 ， 因 为 那个 类 是 从 
Blockable 衍 生出 来 的 。 


在 Blockable 中 ， 变 动 指示 符 是 一 个 int i; 衍生 类 的 run0) 方 法 会 为 其 增 
值 o 


针对 每 个 Bloackable 对 象 ， 都 会 启动 Peeker 类 的 一 个 线程 。Peeker 的 任务 
是 调用 read0 方 法 ， 检 查 与 自己 关联 的 Blockable 对 象 ， 看 看 i 是 否 发 生 了 
变化 ， 最 后 用 它 的 status 文 本 字段 报告 检查 结果 。 注 意 read() 和 update() 都 
是 同步 的 ， 要 求 对 象 的 锁定 能 上 自由 解除 ， 这 一 点 非常 重要 。 








1. 睡眠 
这 个 程序 的 第 一 项 测试 是 用 sleepO0 作 出 的 : 


///:Continuing 


///////////// Blocking via sleep() /////////// 
class Sleeper1 extends Blockable { 
public Sleeperi(Container c) { super(c); } 
public synchronized void run() { 
while(true) { 
i++; 
update(); 
try { 


sleep(1000); 


} catch (InterruptedException e){} 


class Sleeper2 extends Blockable { 
public Sleeper2(Container c) { super(c); } 
public void run() { 
while(true) { 


change(); 


try { 
sleep(1000); 


} catch (InterruptedException e){} 


} 
public synchronized void change() { 
工 + 十 ， 


了 


update(); 
} 


} ///:Continued 





在 Sleeper1 中 ， 整 个 run() 方 法 都 是 同步 的 。 我 们 可 看 到 与 这 个 对 象 关联 
在 一 起 的 Peeker 可 以 正常 运行 ， 直 到 我 们 启动 线程 为 止 ， 随 后 Peeker 便 
会 完全 停止 。 这 正 是 “堵塞 ”的 一 种 形式 : 因为 Sleeperl.run0 是 同步 的 ， 
而 且 一 旦 线程 启动 ， 它 就 肯定 在 run0 内 部 ， 方 法 永远 不 会 放弃 对 象 锁 
定 ， 造 成 Peeker 线 程 的 堵塞 。 

Sleeper2 通 过 设置 不 同步 的 运行 ， 提 供 了 一 种 解决 方案 。 只 有 change() 方 
法 才 是 同步 的 ， 所 以 尽管 run0 位 于 sleepO 内 部 ，Peeker 仍 然 能 访问 上 自己 
需要 的 同步 方法 一 一 read()。 在 这 里 ， 我 们 可 看 到 在 启动 了 Sleeper2 线 程 
以 后 ，Peeker 会 持续 运行 下 去 。 


2. 暂 集 和 恢复 


这 个 例子 接 下 来 的 一 部 分 引入 了 “ 挂 起 ”或 者 “暂停 ”(Suspend) 的 概述 。 
Thread 类 提供 了 一 个 名 为 suspend() 的 方法 ， 可 临时 中 止 线程 ， 以 及 一 个 
名 为 resume() 的 方法 ， 用 于 从 和 暂停 处 开始 恢复 线程 的 执行 。 显 然 ， 我 们 
可 以 推断 出 resume0O 是 由 暂停 线程 外 部 的 某 个 线程 调用 的 。 在 这 种 情况 
下 ， 需 要 用 到 一 个 名 为 Resumer〔 恢 复 器 〉 的 独立 类 。 演 示 和 暂停 / 恢复 
过 程 的 每 个 类 都 有 一 个 相关 的 恢复 器 。 如 下 所 示 : 


























///:Continuing 


/////////// Blocking via suspend() /////////// 
class SuspendResume extends Blockable { 
public SuspendResume(Container c) { 
super(c); 


new Resumer(this); 


} 


class SuspendResumei extends SuspendResume { 
public SuspendResume1(Container c) { super(c);} 
public synchronized void run() { 
while(true) { 
i++; 


update(); 


suspend(); // Deprecated in Java 1.2 


} 


class SuspendResume2 extends SuspendResume { 
public SuspendResume2(Container c) { super(c);} 
public void run() { 


while(true) { 


change(); 


suspend(); // Deprecated in Java 1.2 


} 
public synchronized void change() { 
i++; 


了 


update(); 


} 


class Resumer extends Thread { 
private SuspendResume sr; 
public Resumer(SuspendResume sr) { 
this.sr = sr; 
start(); 
} 
public void run() { 
while(true) { 
try { 
sleep(1000); 
} catch (InterruptedException e){} 


sr.resume(); // Deprecated in Java 1.2 


} ///: Continued 


SuspendResume1 也 提供 了 一 个 同步 的 run0) 方 法 。 同 样 地 ， 当 我 们 启动 这 
个 线程 以 后 ， 就 会 发 现 与 它 关 联 的 Peeker 进 入 堵塞" 状态， 等候 对 象 锁 
被 释放 ， 但 那 永远 不 会 发 生 。 和 往 弟 一 样 ， 这 个 问题 在 SuspendResume2 
里 得 到 了 解决 ， 它 并 不 同步 整个 mn() 方 法 ， 而 是 采用 了 一 个 单独 的 同步 
change() 方 法 。 


对 于 Java 1.2， 大 家 应 注意 suspend0 和 resumeO 已 获得 强烈 反对 ， 因 为 
suspend), SWRA, POA HEA. haz, RAD 
会 看 到 许多 被 锁 住 的 对 象 在 傻乎乎 地 等 竺 对方。 这 会 造成 整个 应 用 程序 
的 “凝固 ?。 尺 管 在 一 些 老 程序 中 还 能 看 到 它们 的 踪迹 ， 但 在 你 写 自己 的 
程序 时 ， 无 论 如 何 都 应 避免 。 本 章 稍 后 就 会 讲述 正确 的 方案 是 什么 。 


3. 等 待 和 通知 


通过 前 两 个 例子 的 实践 ， 我 们 知道 无 论 sleepO 还 是 suspend0O 都 不 会 在 目 

己 被 调用 的 时 候 解 除 锁定 。 需 要 用 到 对 象 锁 时 ， 请 务必 注意 这 个 问题 。 

在 男 一 方面 ，wait() 方 法 在 被 调用 时 却 会 解除 锁定 ， 这 意味 着 可 在 执行 

wait() 期 间 调 用 线程 对 象 中 的 其 他 同步 方法 。 但 在 接着 的 两 个 类 中 ， 我 

们 看 到 run0) 方 法 都 是 “同步 ”的 。 在 waitO 期 间 ，Peeker 仍 然 拥 有 对 同步 方 
这 是 由 于 wait0 在 挂 起 内 部 调用 的 方法 时 ， 会 解除 

对 象 的 锁定 。 


我 们 也 可 以 看 到 waitO 的 两 种 形式 。 第 一 种 形式 采用 一 个 以 蜡 秒 为 单位 
的 参数 ， 它 共有 与 sleepO0 中 相同 的 含义 : 暂停 这 一 段 规 定时 间 。 区 别 在 
于 在 wait0 中 ， 对 象 锁 已 被 解除 ， 而 且 能 够 自由 地 退出 wait0， 因 为 一 个 
notify() 可 强行 使 时 间 流 挝 。 


第 二 种 形式 不 采用 任何 参数 ， 这 意味 着 wait0 会 持续 执行 ， 直 到 notify0 
介入 为 止 。 而 且 在 一 段 时 间 以 后 ， 不 会 目 行 中 止 。 


wait() 和 notifyO 比 较 特 别 的 一 个 地 方 是 这 两 个 方法 都 属于 基础 类 Object 的 
一 部 分 ， 不 象 sleep()，suspend() 以 及 resume() 那 样 属于 Thread 的 一 部 分 。 
尽管 这 表面 看 有 点 儿 奇 怪 一 一 居然 让 专门 进行 线程 处 理 的 东西 成 为 通用 
基础 类 的 一 部 分 一 一 但 仔细 想 想 又 会 释然 ， 因 为 它们 操纵 的 对 象 锁 也 属 
于 每 个 对 象 的 一 部 分 。 因 此 ， 我 们 可 将 一 个 wait0 置 入 任何 同步 方法 内 











部 ， 无 论 在 那个 类 里 是 否 准 备 进 行 涉及 线程 的 处 理 。 事 实 上 ， 我 们 能 调 
用 wait0) 的 唯一 地 方 是 在 一 个 同步 的 方法 或 代码 块 内 部 。 寿 在 一 个 不 同 

步 的 方法 内 调用 wait0 或 者 notify0， 尽 管 程序 仍然 会 编译 ， 但 在 运行 它 

的 时 候 ， 就 会 得 到 一 个 llegalMonitorStateException 〈 非 法 监视 器 状态 违 
例 ) ， 而 且 会 出 现 多 少 有 点 莫名 其 妙 的 一 条 消息 : “current thread not 
owner”( 当 前 线程 不 是 所 有 人 ”。 注 意 sleep()，suspend() 以 及 resume() 都 

能 在 不 同步 的 方法 内 调用 ， 因 为 它们 不 需要 对 锁定 进行 操作 。 


只 能 为 自己 的 锁定 调用 wait0 和 notify0。 同 样 地 ， 仍 然 可 以 编译 那些 试 
图 使 用 错误 锁定 的 代码 ， 但 和 往常 一 样 会 产生 同样 的 
IllegalMonitorStateException 违 例 。 我 们 没 办 法 用 其 他 人 的 对 象 锁 来 感 弄 
系统 ， 但 可 要 求 男 一 个 对 象 执行 相应 的 操作 ， 对 它 自 己 的 锁 进 行 操作 。 
所 以 一 种 做 法 是 创建 一 个 同步 方法 ， 令 其 为 自己 的 对 象 调用 notify()。 但 
在 Notifier 中 ， 我 们 会 看 到 一 个 同步 方法 内 部 的 notify0: 


synchronized(wn2) { 








wn2.notify(); 


其 中 ，wn2 是 类 型 为 WaitNotify2 的 对 象 。 尽 管 并 不 属于 WaitNotify2 的 一 
部 分 ， 这 个 方法 仍然 获得 了 wn2 对 象 的 锁定 。 在 这 个 时 候 ， 它 为 wn2 调 
用 notify0O 是 合法 的 ， 不 会 得 到 IlegalMonitorStateException 违 例 。 





///:Continuing 


/////////// Blocking via wait() /////////// 
class WaitNotifyi extends Blockable { 
public WaitNotify1(Container c) { super(c); } 
public synchronized void run() { 
while(true) { 


i++; 


了 


update(); 
try { 
wait(1000); 


} catch (InterruptedException e){} 


} 
class WaitNotify2 extends Blockable { 
public waitNotify2(Container c) { 
super(c); 
new Notifier(this); 
} 
public synchronized void run() { 
while(true) { 
i++; 
update(); 
try { 
wait(); 


} catch (InterruptedException e){} 


} 


class Notifier extends Thread { 


private WaitNotify2 wn2; 
public Notifier(WaitNotify2 wn2) { 
this.wn2 = wn2; 
start(); 
} 
public void run() { 
while(true) { 
try { 
sleep(2000); 
} catch (InterruptedException e){} 
synchronized(wn2) { 


wn2.notify(); 


} 


} ///:Continued 


若 必 须 等 候 其 他 某 些 条 件 〈( 从 线程 外 部 加 以 控制 ) 发 生变 化 ， 同 时 又 不 
想 在 线程 内 一 直 伊 乎 平地 等 下 去 ， 一 般 就 需要 用 到 wait()。wait0 允 许 我 
们 将 线程 置 入 “睡眠 ?状态 ， 同 时 又 “积极 ”地 等 待 条件 发 生 改 变 。 而 且 只 
有 在 一 个 notify0O 或 notifyA10O 发 生变 化 的 时 候 ， 线 程 才 会 被 唤醒 ， 并 检 
查 条 件 是 否 有 变 。 因 此 ， 我 们 认为 它 提供 了 在 线程 间 进 行 同步 的 一 种 手 


段 。 








4. IO 堵塞 
敬一 个 数据 流 必 须 等 候 一 些 IO 活 动 ， 便 会 自动 进入 “堵塞” 状态 。 在 本 例 


下 面 列 出 的 部 分 中 ， 有 两 个 类 协同 通用 的 Reader 以 及 Writer 对 象 工作 
(使 用 Java 1.1 的 流 ) 。 但 在 测试 模型 中 ， 会 设置 一 个 管道 化 的 数据 
使 两 个 线程 相互 间 能 安全 地 传递 数据 (这 正 是 使 用 管道 流 的 目 
J)o 


Sender 将 数据 置 入 Writer， 并 “睡眠 ”随机 长 短 的 时 间 。 然 而 ，Receiver 本 
身 并 没有 包括 sleepO0，suspend0 或 者 wait0) 方 法 。 但 在 执行 read0 的 时 
候 ， 如 果 没 有 数据 存在 ， 它 会 自动 进入 “ 墙 塞 ”状态 。 如 下 所 示 : 


///:Continuing 








class Sender extends Blockable { // send 
private Writer out; 
public Sender(Container c, Writer out) { 
super(c); 
this.out = out; 
} 
public void run() { 
while(true) { 
for(char c = 'A'; c <= 'z'; c++) { 
try { 
i++; 
out.write(c); 
state.setText("Sender sent: " 
+ (char)c); 
sleep((int)(3000 * Math.random())); 


} catch (InterruptedException e){} 


catch (IOEXception e) {} 


} 


class Receiver extends Blockable { 
private Reader in; 
public Receiver(Container c, Reader in) { 
super(c); 
this.in = in; 
} 
public void run() { 
try { 
while(true) { 
i++; // Show peeker it's alive 
// Blocks until characters are there: 
state.setText("Receiver read: " 


+ (char)in.read()); 


} 


} catch(IOException e) { e.printStackTrace();} 
} 


} ///:Continued 


这 两 个 类 也 将 信息 送 入 目 己 的 state 字 段 ， 并 修改 i 值 ， 使 Peeker 知 道 线程 
仍 在 运行 。 


5. 测试 


令 人 惊讶 的 是 ， 主 要 的 程序 片 (Applet)〉 类 非常 简单 ， 这 是 大 多 数 工 作 
都 已 置 入 Blockable 框 架 的 缘故 。 大 概 地 说 ， 我 们 创建 了 一 个 由 Blockable 
对 象 构成 的 数组 。 而 且 由 于 每 个 对 象 都 是 一 个 线程 ， 所 以 在 按 

下 “start” 按 钮 后 ， 它 们 会 采取 上 自己 的 行动 。 还 有 男 一 个 按钮 和 
actionPerformed() 从 句 ， 用 于 中 止 所 有 Peeker 对 象 。 由 于 Java LI 
对 ”使 用 Thread 的 stop(0 方 法 ， 所 以 可 考虑 采用 这 种 折衷 形式 的 中 止 方 


TK 0 


为 了 在 Sender 和 Receiver 之 间 建 立 一 个 连接 ， 我 们 创建 了 一 个 
PipedWriter 和 一 个 PipedReader。 注 意 PipedReader in 必 须 通 过 一 个 构建 器 
参数 同 PipedWriterout 连 接 起 来 。 在 那 以 后 ， 我 们 在 out 内 放 进 去 的 所 有 
东西 都 可 从 hn 中 提取 出 来 似乎 那些 东西 是 通过 一 个 “管道 ”传输 过 去 
的 。 随 后 将 im 和 out 对 象 分 别传 递 给 Receiver 和 Sender 构 建 器 ;后 者 将 它 
们 当 作 任意 类 型 的 Reader 和 Writer 看 待 ( 也 就 是 说 ， 它 们 被 “上 济 ” 造 型 
TT 


Blockable 句 柄 b 的 数组 在 定义 之 初 并 未 得 到 初始 化 ， 因 为 管道 化 的 数据 
流 是 不 可 在 定义 前 设置 好 的 〈 对 try 块 的 需要 将 成 为 障碍 ) : 


///:Continuing 














/////////// Testing Everything /////////// 
public class Blocking extends Applet { 
private Button 
start = new Button("Start"), 
stopPeekers = new Button("Stop Peekers"); 
private boolean started = false; 


private Blockable[] b; 


private Pipedwriter out; 
private PipedReader in; 
public void init() { 
out = new Pipedwriter(); 
try { 
in = new PipedReader (out); 
} catch(IOException e) {} 
b = new Blockable[] { 
new Sleeperi(this), 
new Sleeper2(this), 
new SuspendResume1(this), 
new SuspendResume2(this), 
new WaitNotify1(this), 
new WaitNotify2(this), 
new Sender(this, out), 
new Receiver(this, in) 
}; 
start.addActionListener(new StartL()); 
add(start); 
stopPeekers.addActionListener ( 
new StopPeekersL()); 


add(stopPeekers); 


class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(!started) { 
started = true; 
for(int i = 0; i < b.length; i++) 


b[i].start(); 


} 


class StopPeekersL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
// Demonstration of the preferred 
// alternative to Thread.stop(): 
for(int i = 0; i < b.length; i++) 


b[i].stopPeeker(); 


} 


public static void main(String[] args) { 
Blocking applet = new Blocking(); 
Frame aFrame = new Frame("Blocking"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 


public void windowClosing(WindowEvent e) { 


System.exit(0); 
} 

}); 
aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(350,550); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 

} 
} //7:~ 


在 init0 中 ， 注 意 循 环 会 过 历 整个 数组 ， 并 为 页 添加 state 和 peeker.status 文 
本 字段 。 


首次 创建 好 Blockable 线 程 以 后 ， 每 个 这 样 的 线程 都 会 自动 创建 并 局 动 自 
己 的 Peeker。 所 以 我 们 会 看 到 各 个 Peeker 都 在 Blockable 线 程 启动 之 前 运 
行 起 来 。 这 一 点 非常 重要 ， 因 为 在 Blockable 线 程 启动 的 时 候 ， 部 分 
Peeker 会 被 堵塞 ， 并 停止 运行 。 弄 懂 这 一 点 ， 将 有 助 于 我 们 加 深 对 “ 墙 
窟 ”这 一 概念 的 认识 。 


14.3.2 死 锁 


由 于 线程 可 能 进入 墙 竖 状态 ， 而 且 由 于 对 象 可 能 拥有 “同步 方 ; 

非 同步 锁定 被 解除 ， 人 否则 线程 不 能 访问 那个 对 象 一 一 所 以 一 个 线程 完 
可 能 等 候 另 一 个 对 象 ， U a 以 此 类 推 。 
这 个 “等 候 ” 链 最 可 怕 的 情形 就 是 进入 封闭 状态 
是 第 一 个 对 象 ! 此 时 ， 所 有 线程 才 会 陷入 无 休止 的 相等 BRE ， 大 家 
都 动弹 不 得 。 我 们 将 这 种 情况 称 为 “ 死 锁 ”。 管 这 种 情况 并 非 经 营 由 
现 ， 但 一 旦 碰 到 ， 程 序 的 调试 将 变 得 异常 艰难 。 


就 语言 本 身 来 说 ， 尚 未 直接 提供 防止 死 锁 的 帮助 措施 ， 需 要 我 们 通过 谱 
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门 可 用 的 。 


1. Java 1.2 对 stop()，suspend()，resume() 以 及 destroy0 的 反对 


为 减少 出 现 死 锁 的 可 能 ，Java 1.2 作 出 的 一 项 贡献 是 “反对 ”使 用 Thread 的 
stop()，suspend()，resume() 以 及 destroy() 方 法 。 


之 所 以 反对 使 用 stopO0， 是 因为 它 不 安全 。 它 会 解除 由 线程 获取 的 所 有 
锁定 ， 而 且 如 果 对 象 处 于 一 种 不 连贯 状态 (“被 破坏 ”) ， 那 么 其 他 线程 
能 在 那 种 状态 下 检查 和 修改 它们 。 结 果 便 造成 了 一 种 微妙 的 局 面 ， 我 们 
很 难 检查 出 真正 的 问题 所 在 。 所 以 应 尽量 避免 使 用 stop()， 应 该 采用 
Blocking.java 那 样 的 方法 ， 用 一 个 标志 告诉 线程 什么 时 候 通 过 退出 自己 
的 run0) 方 法 来 中 止 目 己 的 执行 。 

如 果 一 个 线程 被 堵塞 ， 比 如 在 它 等 候 输入 的 时 候 ， 那 么 一 般 都 不 能 象 在 
Blocking.java 中 那样 轮 询 一 个 标志 。 但 在 这 些 情况 下 ， 我 们 仍然 不 该 使 
用 ， 而 应 换 用 由 Thread 提 供 的 interruptO 方 法 ， 以 便 中 止 并 退出 堵 
塞 的 代码 。 


//: Interrupt.java 




















// The alternative approach to using stop() 
// when a thread is blocked 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
class Blocked extends Thread { 
public synchronized void run() { 


try { 


wait(); // Blocks 


} catch(InterruptedException e) { 
System.out.printin("InterruptedException" ); 
} 


System.out.println("Exiting run()"); 


} 
public class Interrupt extends Applet { 
private Button 
interrupt = new Button("Interrupt"); 
private Blocked blocked = new Blocked(); 
public void init() { 
add(interrupt); 
interrupt .addActionListener ( 
new ActionListener() { 
public 
void actionPerformed(ActionEvent e) { 
System.out.printin("Button pressed"); 
if(blocked == null) return; 
Thread remove = blocked; 
blocked = null; // to release it 


remove.interrupt(); 


}); 


blocked.start(); 
} 
public static void main(String[] args) { 
Interrupt applet = new Interrupt(); 
Frame aFrame = new Frame("Interrupt"); 
aFrame.addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(200, 100); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
} ///:~ 


Blocked.run0 内 部 的 waitO 会 产生 堵 寄 的 线程 。 当 我 们 按 下 按钮 以 后 ， 
blocked (堵塞) 的 句柄 就 会 设 为 null， 使 垃圾 收集 器 能 够 将 其 清除 ， 然 
后 调用 对 象 的 interrupt0 方 法 。 如 果 是 首次 按 下 按钮 ， 我 们 会 看 到 线程 正 
和 退出 。 但 在 没有 可 供 “ 杀 死 ” 的 线程 以 后 ， 看 到 的 便 只 是 按钮 被 按 下 而 
ae 





suspend0 和 resume() 方 法 天 生 容 易 发 生死 锁 。 调 用 suspendO 的 时 候 ， 目 
标 线程 会 停 下 来 ， 但 却 仍然 持 有 在 这 之 前 获得 的 锁定 。 此 时 ， 其 他 任何 
线程 都 不 能 访问 锁定 的 资源 ， 除 非 被 “ 挂 起 ”的 线程 恢复 运行 。 对 任何 线 
程 来 说 ， 如 果 它 们 想 恢复 目标 线程 ， 同 时 又 试图 使 用 任何 一 个 锁定 的 资 
源 ， 就 会 造成 令 人 难堪 的 死 锁 。 所 以 我 们 不 应 该 使 用 suspend() 和 
resume()， 而 应 在 自己 的 Thread 类 中 置 入 一 个 标志 ， 指 出 线程 应 该 活动 
还 是 挂 起 。 寿 标志 指出 线程 应 该 挂 起 ， 便 用 wait0 命 其 进入 等 待 状态 。 
若 标志 指出 线程 应 当 恢复 ， 则 用 一 个 notify0 重 新 启动 线程 。 我 们 可 以 修 
改 前 面 的 Counter2.java 来 实际 体验 一 番 。 尽 管 两 个 版 本 的 效果 是 差不多 
的 ， 但 大 家 会 注意 到 代码 的 组 织 结构 发 生 了 很 大 的 变化 一 一 为 所 有 “上 听 
众 ” 都 使 用 了 匿名 的 内 部 类 ， 而 且 Thread 是 一 个 内 部 类 。 这 使 得 程序 的 
ee 因为 它 取消 了 Counter2.java 中 一 些 额 外 的 记录 工 

















//: Suspend.java 


// The alternative approach to using suspend( ) 
// and resume(), which have been deprecated 
// in Java 1.2. 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
public class Suspend extends Applet { 
private TextField t = new TextField(10); 
private Button 
suspend = new Button("Suspend"), 
resume = new Button("Resume") ; 
class Suspendable extends Thread { 


private int count = 0; 


private boolean suspended = false; 
public Suspendable() { start(); } 
public void fauxSuspend() { 
suspended = true; 
} 
public synchronized void fauxResume() { 
suspended = false; 
notify(); 
} 
public void run() { 
while (true) { 
try { 
sleep(100); 
synchronized(this) { 
while( suspended) 
wait(); 
} 
} catch (InterruptedException e){} 


t.setText(Integer.toString(count++) ); 


} 


private Suspendable ss = new Suspendable(); 


public void init() { 
add(t); 
suspend.addActionListener ( 
new ActionListener() { 
public 
void actionPerformed(ActionEvent e) { 
ss.fauxSuspend(); 
} 


}); 
add(suspend); 


resume.addActionListener ( 
new ActionListener() { 
public 
void actionPerformed(ActionEvent e) { 
ss.fauxResume(); 
} 
}); 
add(resume); 
} 
public static void main(String[] args) { 
Suspend applet = new Suspend(); 
Frame aFrame = new Frame("Suspend"); 


aFrame.addWindowListener ( 


new WindowAdapter() { 
public void windowClosing(WindowEvent e){ 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(300, 100); 
applet.init(); 
applet.start(); 


aFrame.setVisible(true); 


} 
} ///:~ 


Suspendable 中 的 suspended (OCHE) 标志 用 于 开关 “ 挂 起 ”或 者 “暂停 ” 状 
态 。 为 挂 起 一 个 线程 ， 只 需 调用 fauxSuspend() 将 标志 设 为 true CA) BP 
可 。 对 标志 状态 的 侦 测 是 在 ran(0) 内 进行 的 。 束 象 本 章 早 些 时 候 提 到 的 那 
样 ，wait0 必 须 设 为 “同步 ”(synchronized) ， 使 其 能 够 使 用 对 象 锁 。 在 
fauxResume(O 中 ，suspended 标 志 被 设 为 false〈 假 ) ， 并 调用 notify0) 
由 于 这 会 在 一 个 “同步 ”从句 中 唤醒 wait0)， 所 以 fauxResume0) 方 法 也 必须 
同步 ， 使 其 能 在 调用 notify0 之 前 取得 对 象 锁 〈 这 样 一 来 ， 对 象 锁 可 由 要 
唤醒 的 那个 waitO 使 用 ) 。 如 果 遵 照 本 程序 展示 的 样式 ， 可 以 避免 使 用 
wait() 和 notify()。 


Thread 的 destroy() 方 法 根本 没有 实现 ， 它 类 似 一 个 根本 不 能 恢复 的 
Suspend0， 上 所 以 会 发 生 与 suspend0 一 样 的 死 锁 问 题 。 然 而 ， 这 一 方法 没 
有 得 到 明确 的 “反对 ”， 也 许 会 在 Java 以 后 的 版 本 《〈1.2 版 以 后 ) 实现 ， 用 
于 一 些 可 以 承受 死 锁 危险 的 特殊 场合 。 


大 家 可 能 会 奇怪 当初 为 什么 要 实现 这 些 现在 又 被 “反对 ”的 方法 。 之 所 以 
会 出 现 这 种 情况 ， 大 概 是 由 于 Sun 公 司 主要 让 技术 人 员 来 决定 对 语言 的 




















改动 ， 而 不 是 那些 市 场 销售 人 员 。 通 第 ， 技 术 人 员 比 搞 销售 的 更 能 理解 
语言 的 实质 。 当 初 犯 下 了 错误 以 后 ， 也 能 较为 理智 地 正视 它们 。 这 意味 
着 Java 能 够 继续 进步 ， 即 便 这 使 Java 程 序 员 多 少 感到 有 些 不 便 。 就 我 自 
己 来 说 ， 宁 愿 面 对 这 些 不 便 之 处 ， 也 不 愿 看 到 语言 停滞 不 前 。 








14.4 优先 级 


线程 的 优先 级 (Priority) 告诉 调试 程序 该 线程 的 重要 程度 有 多 大 。 如 果 
有 大 量 线程 都 被 堵 蹇 ， 都 在 等 候 运行 ， 调 试 程 序 会 首先 运行 具有 最 高 优 
先 级 的 那个 线程 。 然 而 ， 这 并 不 表示 优先 级 较 低 的 线程 不 会 运行 〈 换 言 
之 ， 不 会 因为 存在 优先 级 而 导致 死 锁 ) 。 知 线程 的 优先 级 较 低 ， 只 不 过 
表示 它 被 准许 运行 的 机 会 小 一 些 而 已 。 

可 用 getPriority(0) 方 法 读 取 一 个 线程 的 优先 级 ， 并 用 setPriority0O 改 变 它 。 
在 下 面 这 个 程序 片 中 ， 大 家 会 发 现 计数 占 的 计数 速度 慢 了 下 来 ， 因 为 它 
们 关联 的 线程 分 配 了 较 低 的 优先 级 : 


//: Counter5 ,java 








// Adjusting the priorities of threads 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
class Ticker2 extends Thread { 
private Button 
b = new Button("Toggle"), 
incPriority = new Button("up"), 
decPriority = new Button("down"); 
private TextField 
t = new TextField(10), 
pr = new TextField(3); // Display priority 


private int count = 0; 


private boolean runFlag = true; 

public Ticker2(Container c) { 
b.addActionListener(new ToggleL()); 
incPriority.addActionListener(new UpL()); 
decPriority.addActionListener(new DownL()); 
Panel p = new Panel(); 
p.add(t); 
p.add(pr); 
p.add(b); 
p.add(incPriority); 
p.add(decPriority); 
c.add(p); 

} 

class ToggleL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 


runFlag = !runFlag; 


i 


Class UpL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
int newPriority = getPriority() + 1; 
if(newPriority > Thread.MAX_PRIORITY) 


newPriority = Thread.MAX_PRIORITY; 


setPriority(newPriority); 


} 


class DownL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
int newPriority = getPriority() - 1; 
if(newPriority < Thread.MIN_PRIORITY ) 
newPriority = Thread.MIN_PRIORITY; 


setPriority(newPriority); 


} 


public void run() { 
while (true) { 
if(runFlag) { 
t.setText(Integer.toString(count++) ); 
pr.setText( 


Integer .toString(getPriority())); 


yield(); 


} 


public class Counter5 extends Applet { 


private Button 


start = new Button("Start"), 


upMax = new Button("Inc Max Priority"), 

downMax = new Button("Dec Max Priority"); 
private boolean started = false; 
private static final int SIZE = 10; 
private Ticker2[] s = new Ticker2[SIZE]; 
private TextField mp = new TextField(3); 
public void init() { 

for(int i = 0; i < s.length; i++) 

s[i] = new Ticker2(this); 


add(new Label("MAX_PRIORITY 


+ Thread.MAX_PRIORITY) ); 


add(new Label("MIN_ PRIORITY 


+ Thread.MIN_PRIORITY) ); 
add(new Label("Group Max Priority = ")); 
add(mp); 
add(start); 
add(upMax); add(downMax) ; 
start.addActionListener(new StartL()); 
upMax.addActionListener(new UpMaxL()); 
downMax.addActionListener (new DownMaxL()); 


showMaxPriority(); 


// Recursively display parent thread groups: 
ThreadGroup parent = 
s[0].getThreadGroup().getParent(); 
while(parent != null) { 
add(new Label ( 
"Parent threadgroup max priority = " 
+ parent.getMaxPriority())); 


parent = parent.getParent(); 


} 


public void showMaxPriority() { 
mp.setText(Integer.toString( 
s[0].getThreadGroup().getMaxPriority())); 
} 
class StartL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
if(!started) { 
started = true; 
for(int i = 0; i < s.length; i++) 


s[i].start(); 


class UpMaxL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
int maxp = 
s[0].getThreadGroup().getMaxPriority(); 
if(++maxp > Thread.MAX_PRIORITY ) 
maxp = Thread.MAX_PRIORITY; 
s[0].getThreadGroup().setMaxPriority(maxp); 


showMaxPriority(); 


} 


class DownMaxL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
int maxp = 
s[0].getThreadGroup().getMaxPriority(); 
if(--maxp < Thread.MIN_PRIORITY) 
maxp = Thread.MIN_PRIORITY; 
s[0].getThreadGroup().setMaxPriority(maxp); 


showMaxPriority(); 


} 


public static void main(String[] args) { 
Counter5 applet = new Counter5(); 


Frame aFrame = new Frame("Counter5"); 


aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(300, 600); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 


TTT 


Ticker 采 用 本 章 前 面 构造 好 的 形式 ， 但 有 一 个 额外 的 TextField〈 文 本 字 
段 ) ， 用 于 显示 线程 的 优先 级 ， 以 及 两 个 额外 的 按钮 ， 用 于 人 为 提高 
降低 优先 级 。 


也 要 注意 yield0 的 用 法 ， 它 将 控制 权 上 自动 返回 给 调试 程序 〈 机 制 ) 。 知 
不 进行 这 样 的 处 理 ， 多 线程 机 制 仍 会 工作 ， 但 我 们 会 发 现 它 的 运行 速度 
慢 了 下 来 〈“ 试 试 删 去 对 yield0 的 调用 ) 。 亦 可 调用 sleepO0， 但 假 铝 那样 

做 ， 计 数 频率 就 会 改 由 sleep() 的 持续 时 间 控 制 ， 而 不 是 优先 级 。 


Counter5 中 的 init0 创 建 了 由 10 个 Ticker2 构 成 的 一 个 数组 ， 它 们 的 按钮 以 
及 输入 字段 (文本 字段 〉 由 Ticker2 构 建 器 置 入 窗 体 。Counter5 增 加 了 新 
的 按钮 ， 用 于 启动 一 切 ， 以 及 用 于 提高 和 降低 线程 组 的 最 大 优先 级 。 除 
此 以 外 ， 还 有 一 些 标签 用 于 显示 一 个 线程 可 以 采用 的 最 大 及 最 小 优先 

级 ; 以 及 一 个 特殊 的 文本 字段 ， 用 于 显示 线程 组 的 最 大 优先 级 〈 在 下 一 
节 里 ， 我 们 将 全 面 讨论 线程 组 的 问题 ) 。 最 后 ， 父 线程 组 的 优先 级 也 作 








为 标签 显示 出 来 。 


按 下 “up”( 上 ) 或 “down”( 下 ) 按钮 的 时 候 ， 会 先 取 得 Ticker2 当 前 的 优 
先 级 ， 然 后 相应 地 提高 或 者 降低 。 


运行 该 程序 时 ， 我 们 可 注意 到 几 件 事情 。 首 先 ， 线 程 组 的 默认 优先 级 是 
5。 即 使 在 局 动 线程 之 前 (或 者 在 创建 线程 之 前 ， 这 要 求 对 代码 进行 适 
Be ee Beda Fea) E Ze 
级 。 


最 简单 的 测试 是 获取 一 个 计数 器 ， 将 它 的 优先 级 降低 至 1， 此 时 应 观察 
到 它 的 计数 频 京 显著 放 慢 。 现 在 试看 再 次 提高 优先 级 ， 可 以 升 高 回 线程 
组 的 优先 级 ， 但 不 能 再 高 了 。 现 在 将 线程 组 的 优先 级 降低 两 次 。 线 程 的 
优先 级 不 会 改变 ， 但 假 知 试图 提高 或 者 降低 它 ， 就 会 发现 这 个 优先 级 目 
动 变 成 线程 组 的 优先 级 。 此 外 ， 新 线程 仍然 具有 一 个 默认 优先 级 ， 即 使 
它 比 组 的 优先 级 还 要 高 〈 换 句 话 说 ， 不 要 指望 利用 组 优先 级 来 防止 新 线 
程 拥 有 比 现 有 的 更 高 的 优先 级 ) 。 


最 后 ， 试 独 提 高 组 的 最 大 优先 级 。 可 以 及 现 ， 这 样 做 是 没有 效果 的 。 我 
们 只 能 减少 线程 组 的 最 大 优先 级 ， 而 不 能 增 大 它 。 


14.4.1 线程 组 


所 有 线程 都 隶属 于 一 个 线程 组 。 那 可 以 是 一 个 默认 线程 组 ， 亦 可 是 一 个 
创建 线程 时 明确 指定 的 组 。 在 创建 之 初 ， 线 程 被 限制 到 一 个 组 里 ， 而 且 
不 能 改变 到 一 个 不 同 的 组 。 每 个 应 用 都 至 少 有 一 个 线程 从 属于 系统 线程 
人 
组 。 


线程 组 也 必须 从 属于 其 他 线程 组 。 必 须 在 构建 器 里 指定 新 线程 组 从 属于 
哪个 线程 组 。 奢 在 创建 一 个 线程 组 的 时 候 没 有 指定 它 的 归属 ， 则 同样 会 
目 动 成 为 系统 线程 组 的 一 名 属 下 。 因 此 ， 一 个 应 用 程序 中 的 所 有 线程 组 
最 终 痢 会 将 系统 线程 组 作为 自己 的 “ 父 ”。 


之 所 以 要 提出 “线程 组 ”的 概念 ， 很 难 从 字面 上 找到 原因 。 这 多 少 为 我 们 
讨论 的 主题 带 来 了 一 些 混乱 。 一 般 地 说 ， 我 们 认为 是 由 于 “安全 ”或 
者 “保密 ”方面 的 理由 才 使 用 线程 组 的 。 根 据 Armold 和 Gosling 的 说 
法 : “线程 组 中 的 线程 可 以 修改 组 内 的 其 他 线程 ， 包 括 那些 位 于 分 层 结 


























构 最 深 处 的 。 一 个 线程 不 能 修改 位 于 自己 所 在 组 或 者 下 属 组 之 外 的 任何 

ARE” CERO) 。 然 而 ， 我 们 很 难 判断 “修改 ”在 这 儿 的 有 具体 含义 是 什 

么 。 下 面 这 个 例子 展示 了 位 于 一 个 “叶子 组 ”内 的 线程 能 修改 它 所 在 线程 

组 树 TR 的 优先 级 ， 同 时 还 能 为 这 个 “ 树 ” 内 的 所 有 线程 都 调用 一 
| 


©: <The Java Programming Language》 第 179 页 。 该 书 由 Arnold 和 Jams 
Gosling 编 著 ，Addison-Wesley 于 1996 年 出 版 





//: TestAccess.java 


// How threads can access other threads 
// in a parent thread group 
public class TestAccess { 
public static void main(String[] args) { 
ThreadGroup 
x = new ThreadGroup("x"), 
y = new ThreadGroup(x, "y"), 
z = new ThreadGroup(y, "z"); 
Thread 


one = new TestThreadi(x, "one"), 


two = new TestThread2(z, "two"); 


class TestThreadi extends Thread { 
private int i; 


TestThreadi(ThreadGroup g, String name) { 


super(g, name); 
} 
void f() { 
i++; // modify this thread 


System.out.println(getName() + " f()"); 


} 


class TestThread2 extends TestThreadi { 
TestThread2(ThreadGroup g, String name) { 
super(g, name); 
start(); 
} 
public void run() { 
ThreadGroup g = 
getThreadGroup().getParent().getParent(); 
g.list(); 
Thread[] gAll = new Thread[g.activeCount()]; 
g.enumerate(gAll1); 
for(int i = 0; i < gAll.length; i++) { 
gAll[i].setPriority(Thread.MIN_PRIORITY); 
((TestThread1)gAl1[i]).(); 


} 


g.list(); 


} 
于 LA 


在 main0 中 ， 我 们 创建 了 几 个 ThreadGroup 〈 线 程 组 ) ， 每 个 都 位 于 不 同 
的 “时 ”上 : x 没有 参数 ， 只 有 它 的 名 字 (一 个 String) ， 所 以 会 自动 进 
入 “system”( 系统) 线程 组 ; y 位 于 x 下 方 ， 而 z 位 于 y 下 方 。 注 意 初始 化 
是 按照 文字 顺序 进行 的 ， 所 以 代码 合法 。 


有 两 个 线程 创建 之 后 进入 了 不 同 的 线程 组 。 其 中 ，TestThread1 没 有 一 个 
run0) 方 法 ， 但 有 一 个 {D， 用 于 通知 线程 以 及 打印 出 一 些 东 西 ， 以 便 我 们 
知道 它 已 被 调用 。 而 TestThread2 属 于 TestThread1 的 一 个 子 类 ， 它 的 run0) 
非常 详尽 ， 要 做 许多 事情 。 首 先 ， 它 获得 当前 线程 所 在 的 线程 组 ， 然 后 
利用 getParent() 在 继承 树 中 向 上 移动 两 级 (这 样 做 是 有 道理 的 ， 因 为 我 
想 把 TestThread2 在 分 级 结构 中 间 下 移动 两 级 ) 。 随 后 ， 我 们 调用 方法 
activeCount()， 人 查询 这 个 线程 组 以 及 所 有 子 线程 组 内 有 多 少 个 线程 ， 从 
而 创建 由 指 问 Thread 的 句 椭 构 成 的 一 个 数组 。enumerate() 方 法 将 指 问 所 
有 这 些 线程 的 句柄 置 入 数组 gAll 里 。 然 后 在 整个 数组 里 遍历 ， 为 每 个 线 
程 都 调用 f0 方 法 ， 同 时 修改 优先 级 。 这 样 一 来 ， 位 于 一 个 “叶子 ”线程 组 
里 的 线程 束 修 改 了 位 于 父 线 程 组 的 线程 。 

调试 方法 list0 打 印 出 与 一 个 线程 组 有 关 的 所 有 信息 ， 把 它们 作为 标准 输 
出 。 在 我 们 对 线程 组 的 行为 进行 调查 的 时 候 ， 这 样 做 是 相当 有 好 处 的 。 
下 面 是 程序 的 输出 : 


java.lang.ThreadGroup[name=x, maxpri=10] 











Thread[one,5,x] 
java.lang.ThreadGroup[name=y, maxpri=10] 
java.lang.ThreadGroup[name=z, maxpri=10] 
Thread[two,5,z] 
one f() 


two f() 


java.lang.ThreadGroup[name=x, maxpri=10] 
Thread[one,1,x] 
java.lang.ThreadGroup[name=y, maxpri=10] 
java.lang.ThreadGroup[name=z, maxpri=10] 


Thread[two,1,z] 


list() 不 仅 打 印 出 ThreadGroup 或 者 Thread 的 类 名 ， 也 打印 出 了 线程 组 的 名 
字 以 及 它 的 最 高 优先 级 。 对 于 线程 ， 则 打印 出 它们 的 名 字 ， 并 接 上 线程 
优先 级 以 及 所 属 的 线程 组 。 注 意 listO0 会 对 线程 和 线程 组 进行 缩 排 处 理 ， 

站 出 它们 是 未 缩 排 的 线程 组 的 “ 子 ”。 


大 家 可 看 到 f() 是 由 TestThread2 的 run() 方 法 调用 的 ， 所 以 很 明显 ， 组 内 的 
所 有 线程 都 是 相当 脆弱 的 。 然 而 ， 我 们 只 能 访问 那些 从 自己 的 system 线 
程 组 树 分 文 出 来 的 线程 ， 而 且 或 许 这 束 是 所 谓 “ 安 全 ”的 意思 。 我 们 不 能 
访问 其 他 任何 人 的 系统 线程 树 。 

1. 线程 组 的 控制 

抛 开 安全 问题 不 谈 ， 线 程 组 最 有 用 的 一 个 地 方 承 是 控制 : 只 需 用 单个 命 
令 即 可 完成 对 整个 线程 组 的 操作 。 下 面 这 个 例子 演示 了 这 一 点 ， 并 对 线 
o 内 优先 级 的 限制 进行 了 说 明 。 括 号 内 的 注释 数字 便于 大 家 比较 输出 


//: ThreadGroup1.java 








// How thread groups control priorities 
// of the threads inside them. 
public class ThreadGroupi { 
public static void main(String[] args) { 
// Get the system thread & print its Info: 


ThreadGroup sys = 


Thread.currentThread().getThreadGroup(); 
sys.list(); // (1) 

// Reduce the system thread group priority: 
sys.setMaxPriority(Thread.MAX_PRIORITY - 1); 
// Increase the main thread priority: 
Thread curr = Thread.currentThread(); 
curr.setPriority(curr.getPriority() + 1); 
sys.list(); // (2) 

// Attempt to set a new group to the max: 
ThreadGroup gi = new ThreadGroup("g1i"); 
gi.setMaxPriority(Thread.MAX_PRIORITY); 

// Attempt to set a new thread to the max: 
Thread t = new Thread(gi, "A"); 
t.setPriority(Thread.MAX_PRIORITY); 
gi.list(); // (3) 

// Reduce gi's max priority, then attempt 

// to increase it: 
gi.setMaxPriority(Thread.MAX_PRIORITY - 2); 
gi.setMaxPriority(Thread.MAX_PRIORITY); 
gi.list(); // (4) 

// Attempt to set a new thread to the max: 

t = new Thread(gi, "B"); 


t.setPriority(Thread.MAX_PRIORITY); 


gi.list(); // (5) 
// Lower the max priority below the default 
// thread priority: 
gi.setMaxPriority(Thread.MIN_PRIORITY + 2); 
// Look at a new thread's priority before 
// and after changing it: 
t = new Thread(g1, "C"); 
gi.list(); // (6) 
t.setPriority(t.getPriority() -1); 
gi.list(); // (7) 
// Make g2 a child Threadgroup of g1 and 
// try to increase its priority: 
ThreadGroup g2 = new ThreadGroup(gi, "g2"); 
g2.list(); // (8) 
g2.setMaxPriority(Thread.MAX_PRIORITY); 
g2.list(); // (9) 
// Add a bunch of new threads to g2: 
for (int i = 0; i < 5; i++) 
new Thread(g2, Integer.toString(1)); 
// Show information about all threadgroups 
// and threads: 
sys.list(); // (10) 


System.out.println( "Starting all threads:"); 


Thread[] all = new Thread[sys.activeCount()]; 
sys.enumerate(all); 
for(int i = 0; i < all.length; i++) 
if(!all[i].isAlive()) 
all[i].start(); 

// Suspends & Stops all threads in 

// this group and its subgroups: 
System.out.printin("All threads started"); 
sys.suspend(); // Deprecated in Java 1.2 

// Never gets here... 
System.out.printin("All threads suspended"); 
sys.stop(); // Deprecated in Java 1.2 
System.out.printin("All threads stopped"); 


} 
NS fief E 


下 面 的 输出 结果 已 进行 了 适当 的 编辑 ， 以 便 用 一 页 能 够 装 下 〈java.lang. 
0 0 
XM: 


(1) ThreadGroup[name=system, maxpri=10] 


Thread[main, 5, system] 
(2) ThreadGroup[name=system, maxpri=9] 


Thread[main,6, system] 


(3) ThreadGroup[name=g1, maxpri=9 ] 
Thread[A,9,g1] 

(4) ThreadGroup[name=g1, maxpri=8 ] 
Thread[A,9,g1] 

(5) ThreadGroup[name=g1, maxpri=8 ] 
Thread[A,9,g1] 
Thread[B, 8,g1] 

(6) ThreadGroup[name=g1, maxpri=3] 
Thread[A,9,g1] 
Thread[B, 8,91] 
Thread[C,6,g1] 

(7) ThreadGroup[name=g1, maxpri=3] 
Thread[A,9,g1] 
Thread[B, 8,91] 
Thread[C,3,g1] 

(8) ThreadGroup[name=g2, maxpri=3] 

(9) ThreadGroup[name=g2, maxpri=3] 

(10)ThreadGroup[name=system, maxpri=9 ] 
Thread[main,6, system] 
ThreadGroup[name=g1, maxpri=3 ] 

Thread[A,9,g1] 
Thread[B, 8,91] 


Thread[C,3,g1] 


ThreadGroup[name=g2, maxpri=3 ] 
Thread[0, 6, g2] 
Thread[1, 6, g2] 

Thread[2, 6,92] 

Thread[3,6,92] 

Thread[4, 6,92] 
Starting all threads: 


All threads started 





所 有 程序 都 至 少 有 一 个 线程 在 运行 ， 而 且 main(0) 采 取 的 第 一 项 行动 便 是 
调用 Thread 的 一 个 static《〈 静 态 ) 方法 ， 名 为 currentThread()。 从 这 个 线 
程 开 始 ， 线 程 组 将 被 创建 ， 而 且 会 为 结果 调用 list()。 输 出 如 下 : 


(1) ThreadGroup[name=system, maxpri=10 | 


Thread[main, 5, system] 
我 们 可 以 看 到 ， 主 线程 组 的 名 字 是 system， 而 主线 程 的 名 字 是 main， 而 
且 它 从 属于 system 线 程 组 。 


第 二 个 练习 显示 出 system 组 的 最 高 优先 级 可 以 减少 ， 而 且 main 线 程 可 以 
增 大 自己 的 优先 级 : 


(2) ThreadGroup[name=system, maxpri=9 ] 


Thread[main,6, system] 


第 三 个 练习 创建 一 个 新 的 线程 组 ， 名 为 g1; 它 上 自动 从 属于 system 线 程 
组 ， 因 为 并 没有 明确 指定 它 的 归属 关系 。 我 们 在 g1 内 部 放置 了 一 个 新 线 
程 ， 名 为 A。 随 后 ， 我 们 试 着 将 这 个 组 的 最 大 优先 级 设 到 最 高 的 级 别 ， 


并 将 A 的 优先 级 也 设 到 最 高 一 级 。 结 果 如 下 : 


(3) ThreadGroup[name=g1, maxpri=9] 


Thread[A,9,g1] 





可 以 看 出 ， 不 可 能 将 线程 组 的 最 大 优先 级 设 为 高 于 它 的 父 线 程 组 。 


第 四 个 练习 将 g1 的 最 大 优先 级 降低 两 级 ， 然 后 试 着 把 它 升 至 
Thread. MAX PRIORITY。 结 果 如 下 : 


(4) ThreadGroup[name=g1, maxpri=8] 


Thread[A,9,g1] 


同样 可 以 看 出 ， 提 高 最 大 优先 级 的 企图 是 失败 的 。 我 们 只 能 降低 一 个 线 
程 组 的 最 大 优先 级 ， 而 不 能 提高 它 。 此 外 ， 注 意 线程 A 的 优先 级 并 未 改 
变 ， 而 且 它 现在 高 于 线程 组 的 最 大 优先 级 。 也 就 是 次， 线程 组 最 大 优先 
级 的 变化 并 不 能 对 现 有 线程 造成 影响 。 

第 五 个 练习 试 着 将 一 个 新 线程 设 为 最 大 优先 级 。 如 下 所 示 : 


(5) ThreadGroup[name=g1, maxpri=8] 








Thread[A,9,g1] 


Thread[B, 8,91] 


因此 ， 新 线程 不 能 变 到 比 最 大 线程 组 优先 级 还 要 高 的 一 级 。 


这 个 程序 的 默认 线程 优先 级 是 6; ATE TERRE, ABE EAU 
先 级 ， 而 且 不 会 发 生变 化 ， 除 非 对 优先 级 进行 了 特别 的 处 理 。 练 习 六 将 
把 线程 组 的 最 大 优先 级 降 至 默认 线程 优先 级 以 下 ， 看 看 在 这 种 情况 下 新 
律 一 个 线程 会 发 生 什么 事情 : 





(6) ThreadGroup[name=g1, maxpri=3 ] 


Thread[A,9,g1] 
Thread[B, 8,91] 


Thread[C,6,g1] 


尽管 线程 组 现在 的 最 大 优先 级 是 3， 但 仍然 用 默认 优先 级 6 来 创建 新 线 
程 。 所 以 ， 线 程 组 的 最 大 优先 级 不 会 影响 默认 优先 级 (事实 上 ， 似 乎 没 
有 办 法 可 以 设置 新 线程 的 默认 优先 级 ) 。 


改变 了 优先 级 后 ， 接 下 来 试 试 将 其 降低 一 级 ， 结 果 如 下 : 


(7) ThreadGroup[name=g1, maxpri=3] 


Thread[A,9,9g1] 
Thread[B,8,g1] 


Thread[C,3,g1] 


F a a a ee, 
限制 。 

我 们 在 (8) 和 (9) 中 进行 了 类 似 的 试验 。 在 这 里 ， 我 们 创建 了 一 个 新 的 线 
旦 组， 名 为 g2， 将 其 作为 g1 的 一 个 子 组 ， 并 改变 了 它 的 最 大 优先 级 。 大 
家 可 以 看 到 ，g2 的 优先 级 无 论 如 何 都 不 可 能 高 于 g1: 


(8) ThreadGroup[name=g2, maxpri=3] 








(9) ThreadGroup[name=g2, maxpri=3] 


也 要 注意 在 g2 创 建 的 时 候 ， 它 会 个 目 动 设 为 g1 的 线程 组 最 大 优先 级 。 


经 过 所 有 这 些 实验 以 后 ， 整 个 线程 组 和 线程 系统 都 会 被 打印 出 来 ， 如 下 
所 示 : 


(10)ThreadGroup[name=system, maxpri=9 | 


Thread[main, 6, system] 
ThreadGroup[name=g1, maxpri=3 | 
Thread[A,9,g1] 
Thread[B, 8,91] 
Thread[C,3,g1] 
ThreadGroup[name=g2, maxpri=3 | 
Thread[0, 6,92] 
Thread[1, 6, g2] 
Thread[2, 6,92] 
Thread[3,6,g2] 


Thread[4,6,g2] 


所 以 由 线程 组 的 规则 所 限 ， 一 个 子 组 的 最 大 优先 级 在 任何 时 候 都 只 能 低 
于 或 等 于 它 的 父 组 的 最 大 优先 级 。 


本 程序 的 最 后 一 个 部 分 演示 了 用 于 整 组 线程 的 方法 。 程 序 首先 过 历 整 个 
线程 树 ， 并 启动 每 一 个 尚未 启动 的 线程 。 例 如 ，system 组 随后 会 被 挂 起 
(暂停 ) ， 最 后 被 中 止 〈 尽 管用 suspend0 和 stopO 对 整个 线程 组 进行 操 
作 看 起 来 似乎 很 有 趣 ， 但 应 注意 这 些 方 法 在 Java 1.2 里 都 是 被 “ 反 
对 ”的 ) 。 但 在 挂 起 system 组 的 同时 ， 也 挂 起 了 main 线 程 ， 而 且 整 个 程 
序 都 会 关闭 。 所 以 永远 不 会 达到 让 线程 中 目的 那 一 步 。 实 际 上 ， 假 如 真 
的 中 止 了 main 线 程 ， 它 会 “ 搓 ? 出 一 个 ThreadDeath 违 例 ， 所 以 我 们 通 稼 不 
这 样 做 。 由 于 ThreadGroup 是 从 Object 继承 的 ， 其 中 包含 了 wait0 方 法 ， 

所 以 也 能 调用 wait( 秒 数 x1000)， 令 程序 暂停 运行 任意 秒 数 的 时 | 间 。 妆 
然 ， 事 前 必须 在 一 个 同步 块 里 取得 对 象 锁 。 














ThreadGroup 类 也 提供 了 suspend0 和 resume(0) 方 法 ， 所 以 能 中 止 和 启动 整 

个 线程 组 和 它 的 所 有 线程 ， 也 能 中 止 和 启动 它 的 子 组 ， 所 有 这 些 只 需 一 
即 可 《〈 再 次 提醒 ，suspend0 和 resume0O 都 是 Java 1.2 所 “ 反 
对 ”的 ) 。 


从 表面 看 ， 线 程 组 似乎 有 些 让 人 换 不 着 头脑 ， 但 请 注意 我 们 很 少 需要 下 
接 使 用 它们 。 





14.5 回顾 runnable 


在 本 章 早 些 时 候 ， 我 兽 建 议 大 家 在 将 一 个 程序 片 或 主 Frame 当 作 
Runnable 的 实现 形式 之 前 ， 一 定 要 好 好 地 想 一 想 。 知 采用 那 种 方式 ， 丈 
只 能 在 自己 的 程序 中 使 用 其 中 的 一 个 线程 。 这 便 限制 了 灵活 性 ， 一 旦 需 
要 用 到 属于 那 种 类 型 的 多 个 线程 ， 就 会 遇 到 不 必要 的 有 抹 烦 。 


当然 ， 如 果 必 须 从 一 个 类 继承 ， 而 且 想 使 类 具有 线程 处 理 能 力 ， 则 
Runnable 是 一 种 正确 的 方案 。 本 章 最 后 一 个 例子 对 这 一 点 进行 了 剖析 ， 
制作 了 一 个 RunnableCanvas 类 ， 用 于 为 自己 描绘 不 同 的 闫 色 (Canvas 
是 “ 男 布 ”的 意思 ) 。 这 个 应 用 被 设计 成 从 命令 行 获得 参数 值 ， 以 决定 闫 
色 网 格 有 多 大 ， 以 及 颜色 发 生变 化 之 间 的 sleepO0 有 多 长 。 通 过 运用 这 些 
值 ， 大 家 能 体验 到 线程 一 些 有 趣 而 且 可 能 令 人 费解 的 特性 : 


//: ColorBoxes. java 











// Using the Runnable interface 
import java.awt.*; 
import java.awt.event.*; 
class CBox extends Canvas implements Runnable { 
private Thread t; 
private int pause; 
private static final Color[] colors = { 
Color.black, Color.blue, Color.cyan, 
Color.darkGray, Color.gray, Color.green, 
Color.lightGray, Color.magenta, 
Color.orange, Color.pink, Color.red, 


Color.white, Color.yellow 


}; 
private Color cColor = newColor(); 
private static final Color newColor() { 
return colors[ 
(int)(Math.random() * colors.length) 
]; 
} 
public void paint(Graphics g) { 
g.setColor(cColor); 
Dimension s = getSize(); 
g.fillRect(0, ©, s.width, s.height); 
} 
public CBox(int pause) { 
this.pause = pause; 
t = new Thread(this); 
t.start(); 
} 
public void run() { 
while(true) { 
cColor = newColor(); 
repaint(); 
try { 


t.sleep(pause); 


} catch(InterruptedException e) {} 
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public class ColorBoxes extends Frame { 
public ColorBoxes(int pause, int grid) { 

setTitle("ColorBoxes"); 

setLayout(new GridLayout(grid, grid)); 

for (int 1 = 0; i < grid * grid; i++) 
add(new CBox(pause)); 

addWindowListener(new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


}); 
} 


public static void main(String[] args) { 
int pause = 50; 
int grid = 8; 
if(args.length > 0) 
pause = Integer.parseInt(args[0]); 
if(args.length > 1) 


grid = Integer.parseInt(args[1]); 


Frame f = new ColorBoxes(pause, grid); 
f.setSize(500, 400); 
f.setVisible(true); 


} 
} ///:~ 


ColorBoxes 是 一 个 典型 的 应 用 (程序 ) ， 有 一 个 构建 器 用 于 设置 GUI。 
这 个 构建 器 采用 int grid 的 一 个 参数 ， 用 它 设置 GridLayout( 网 格 布 
局 )， 使 每 一 维 里 都 有 一 个 grid 和 单元 。 随 后 ， 它 添加 适当 数量 的 CBox 对 
象 ， 用 它们 填充 网 格 ， 并 为 每 一 个 都 传递 pause 值 。 在 main0 中 ， 我 们 可 
看 到 如 何 对 pause 和 grid 的 默认 值 进 行 修改 〈 如 果 用 命令 行 参数 传递 ) 。 


CBox 是 进行 正式 工作 的 地 方 。 它 是 从 Canvas 继 承 的 ， 并 实现 了 Runnable 
接口 ， 使 每 个 Canvas 也 能 是 一 个 Thread。 记 住 在 实现 Runnable 的 时 候 ， 
并 没有 实际 产生 一 个 Thread 对 象 ， 只 是 一 个 拥有 run() 方 法 的 类 。 因 此 ， 
我 们 必须 明确 地 创建 一 个 Thread 对 象 ， 并 将 Runnable 对 象 传 递 给 构建 
人 (在 构建 器 里 进行 )。 在 CBox 里 ， 这 个 线程 的 名 字 
HUTE to 


请 留意 数组 colors， 它 对 Color 类 中 的 所 有 颜色 进行 了 列举 〈 枚 举 ) 。 它 
在 newColor(0 中 用 于 产生 一 种 随机 选择 的 颜色 。 当 前 的 单元 〈 格 ) 颜色 


是 cColor。 


paintO 则 相当 人 简单 
张 画 布 (Canvas) 。 


在 run0 中 ， 我 们 看 到 一 个 无 限 循环 ， 它 将 cColor 设 为 一 种 随机 颜色 ， 然 
后 调用 repaint() 把 它 显 示 出 来 。 随 后 ， 对 线程 执行 sleep()， 使 其 “休眠 ”由 
命令 行 指定 的 时 间 长 度 。 

由 于 这 种 设计 方案 非常 灵活 ， 而 且 线 程 处 理 同 每 个 Canvas 元 素 都 紧密 结 
合 在 一 起 ， 所 以 在 理论 上 可 以 生成 任意 多 的 线程 〈 但 在 实际 应 用 中 ， 这 
要 受到 JVM 能 够 从 容 对 付 的 线程 数量 的 限制 ) 。 


这 个 程序 也 为 我 们 提供 了 一 个 有 趣 的 评测 基准 ， 因 为 它 揭示 了 不 同 JVM 








只 是 将 颜色 设 为 cColor， 然 后 用 那 种 颜色 填充 整 








机 制 在 速度 上 造成 的 戏剧 性 的 差异 。 
14.5.1 过 多 的 线程 


有 些 时 候 ， 我 们 会 发 现 ColorBoxes 几 乎 陷于 停顿 状态 。 在 我 自己 的 机 器 
上 ， 这 一 情况 在 产生 了 10x10 的 网 格 之 后 发 生 了 。 为 什么 会 这 样 呢 ? A 
然 地 ， 我 们 有 理由 怀疑 AWT 对 它 做 了 什么 事情 。 所 以 这 里 有 一 个 例子 
能 够 测试 那个 猜测 ， 它 产生 了 较 少 的 线程 。 代 码 经 过 了 重新 组 织 ， 使 一 
个 Vector 实现 了 Runnable， 而 且 那 个 Vector 容纳 了 数量 众多 的 色 块 ， 并 
随机 挑选 一 些 进行 更 新 。 随 后 ， 我 们 创建 大 量 这 些 Vector 对 象 ， 数 量 大 
致 取决 于 我 们 挑选 的 网 格 维 数 。 结 果 便 是 我 们 得 到 比 色 块 少 得 多 的 线 
程 。 所 以 假如 有 一 个 速度 的 加 快 ， 我 们 就 能 立即 知道 ， 因 为 前 例 的 线程 
数量 太 多 了 。 如 下 所 示 : 


//: ColorBoxes2.java 





// Balancing thread use 
import java.awt.*; 
import java.awt.event.*; 
import java.util.*; 
class CBox2 extends Canvas { 
private static final Color[] colors = { 
Color.black, Color.blue, Color.cyan, 
Color.darkGray, Color.gray, Color.green, 
Color.lightGray, Color.magenta, 
Color.orange, Color.pink, Color.red, 
Color.white, Color.yellow 
}; 


private Color cColor = newColor(); 


private static final Color newColor() { 

return colors[ 
(int)(Math.random() * colors.length) 

]; 

} 

void nextColor() { 
cColor = newColor(); 
repaint(); 

} 

public void paint(Graphics g) { 
g.setColor(cColor); 
Dimension s = getSize(); 


g.fillRect(0, ©, s.width, s.height); 


} 


class CBoxVector 
extends Vector implements Runnable { 
private Thread t; 
private int pause; 
public CBoxVector(int pause) { 
this.pause = pause; 


t = new Thread(this); 


public void go() { t.start(); } 
public void run() { 
while(true) { 
int 1 = (int)(Math.random() * size()); 
((CBox2)elementAt(i)).nextColor(); 
try { 
t.sleep(pause) ; 


} catch(InterruptedException e) {} 
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public class ColorBoxes2 extends Frame { 
private CBoxVector[] v; 
public ColorBoxes2(int pause, int grid) { 
setTitle("ColorBoxes2"); 
setLayout(new GridLayout(grid, grid)); 
v = new CBoxVector[grid]; 
for(int i = 0; i < grid; i++) 
v[i] = new CBoxVector (pause); 
for (int i = 0; i < grid * grid; i++) { 
v[i % grid].addElement(new CBox2()); 


add( (CBox2)v[i % grid].lastElement()); 


for(int i = 0; i < grid; i++) 
v[i].go(); 

addWindowListener(new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


}); 


public static void main(String[] args) { 
// Shorter default pause than ColorBoxes: 
int pause = 5; 

int grid = 8; 
if (args.length > 0) 

pause = Integer.parseInt(args[0]); 
if(args.length > 1) 

grid = Integer.parseInt(args[1]); 
Frame f = new ColorBoxes2(pause, grid); 
f.setSize(500, 400); 
f.setVisible(true); 


} 
Etri 


在 ColorBoxes2 中 ， 我 们 创建 了 CBoxVector 的 一 个 数组 ， 并 对 其 初始 


化 ， 使 其 容 下 各 个 CBoxVector 网 格 。 每 个 网 格 都 知道 自己 该 “睡眠 ”多 长 
的 时 间 。 随 后 为 每 个 CBoxVector 都 添加 等 量 的 Cbox2 对 象 ， 而 且 将 每 个 
Vector 都 告诉 给 go0， 用 它 来 司 动 自己 的 线程 。 


CBox2 类 似 CBox 能 用 一 种 随机 选择 的 颜色 描绘 自己 。 但 那 就 是 
CBox2 能 够 做 的 全 部 工作 。 所 有 涉及 线程 的 处 理 都 已 移 至 CBoxVector 进 
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CBoxVector 也 可 以 拥有 继承 的 Thread， 并 有 一 个 类 型 为 Vector 的 成 员 对 
象 。 这 样 设 计 的 好 处 就 是 addFlement0 和 elementAt0 方 法 可 以 获得 特定 
的 参数 以 及 返回 值 类 型 ， 而 不 是 只 能 获得 种 规 Object《〈 它 们 的 名 字 也 可 
以 变 得 更 短 ) 。 然 而 ， 这 里 采用 的 设计 表面 上 看 需要 较 少 的 代码 。 除 此 
以 外 ， 它 会 目 动 保留 一 个 Vector 的 其 他 所 有 行为 。 由 于 elementAt() 需 要 
大 量 进行 “封闭 ”工作 ， 用 到 许多 括号 ， 所 以 随 着 代码 主体 的 扩充 ， 最 终 
仍 有 可 能 需要 大 量 代码 。 


和 以 前 一 样 ， 在 我 们 实现 Runnable 的 时 候 ， 并 没有 获得 与 Thread 配 套 提 
供 的 所 有 功能 ， 所 以 必须 创建 一 个 新 的 Thread， 并 将 上 自己 传递 给 它 的 构 
建 咽 ， 以 便 正式 “启动 > 一 一 start() 一 一 一 些 东 西 。 大 家 在 CBoxVector 构 
建 器 和 go0 里 都 可 以 体会 到 这 一 点 。run0) 方 法 简单 地 选择 Vector 里 的 一 
as 并 为 那个 元 素 调用 nextColor()， 令 其 挑选 一 种 新 的 随 
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运行 这 个 程序 时 ， 大 家 会 发 现 它 确实 变 得 更 快 ， 啊 应 也 更 迅速 《比如 在 
中 断 它 的 时 候 ， 它 能 更 快 地 停 下 来 ) 。 而 且 随 着 网 格 尺寸 的 壮大 ， 它 也 
不 会 经 常 性 地 陷于 “人 停顿? 状态。 因此， 线程 的 处 理 又 多 了 一 项 新 的 考虑 
因素 : 必须 随时 检查 目 己 有 没有 “ 太 多 的 线程 ”〈 无 论 对 什么 程序 和 运行 
FE) 。 知 线程 太 多 ， 必 须 试 着 使 用 上 面 介 绍 的 技术 ， 对 程序 中 的 线程 
数量 进行 “平衡 ?>。 如 果 在 一 个 多 线程 的 程序 中 遇 到 了 性 能 上 的 问题 ， 那 
么 现在 有 许多 因素 需要 检查 : 


(1) 对 sleep，yield0 以 及 /或 者 waitO 的 调用 足够 多 吗 ? 
(2) sleepO 的 调用 时 间 足 够 长 吗 ? 
(3) 运行 的 线程 数 是 不 是 太 多 ? 
(4) 试 过 不 同 的 平台 和 JVM 吗 ? 















































象 这 样 的 一 些 问题 是 造成 多 线程 应 用 程序 的 编制 成 为 一 种 “技术 活 ” 的 原 
AlZ—. 


14.6 总 结 


何 时 使 用 多 线程 技术 ， 以 及 何 时 避免 用 它 ， 这 是 我 们 需要 掌握 的 重要 课 
题 。 骼 它 的 主要 目的 是 对 大 量 任务 进行 有 序 的 管理 。 通 过 多 个 任务 的 混 
合 使 用 ， 可 以 更 有 效 地 利用 计算 机 资源 ， 或 者 对 用 户 来 说 显得 更 方便 。 
资源 均衡 的 经 典 问 题 是 在 IO 等 候 期 间 如 何 利 用 CPU。 至 于 用 户 方面 的 方 
便 性 ， 最 经 典 的 问题 就 是 如 何在 一 个 长 时 间 的 下 载 过 程 中 监视 并 灵敏 地 
反应 一 个 “停止 ” (stop) 按钮 的 按 下 。 


多 线程 的 主要 缺点 包括 : 
(1) 等 候 使 用 共享 资源 时 造成 程序 的 运行 速度 变 慢 。 
(2) 对 线程 进行 管理 要 求 的 额外 CPU 开 销 。 


(3) ”复杂 程度 无 意义 的 加 大 ， 比 如 用 独立 的 线程 来 更 新 数组 内 每 个 元 素 
MARRET. 


(4) INE. IRB ZI E E PARKIEM EE ER RETE TK o 


线程 为 一 个 优点 是 它们 用 “ 轻 度 ”执行 切换 (100 条 指令 的 顺序 ) 取代 

了 “重度 ”进程 场景 切换 (1000 条 指令 ) 。 由 于 一 个 进程 内 的 所 有 线程 共 
译 相 同 的 内 存 空间 ， 所 以 “ 轻 度 ”场景 切换 只 改变 程序 的 执行 和 本 地 变 
量 。 而 在 “重度 ?场景 切换 时 ， 一 个 进程 的 改变 要 求 必须 完整 地 交换 内 存 
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线程 处 理 看 来 好 象 进 入 了 一 个 全 新 的 领域 ， 似 乎 要 求 我 们 学 习 一 种 全 新 
的 程序 设计 语言 一 一 或 者 至 少 学 习 一 系列 新 的 语言 概念 。 由 于 大 多 数 微 
机 操作 系统 者 提供 了 对 线程 的 文 持 ， 所 以 程序 设计 语言 或 者 库 里 也 出 现 
了 对 线程 的 扩展 。 不 管 在 什么 情况 下 ， 涉 及 线程 的 程序 设计 : 


(1) 刚 开 始 会 让 人 措 不 着 头脑 ， 要 求 改换 我 们 传统 的 编程 思路 ; 


2) 。 其 他 语言 对 线程 的 支持 看 来 是 类 似 的 。 所 以 一 旦 掌握 了 线程 的 概 

念 ， 在 其 他 环境 也 不 会 有 太 大 的 困难 。 尽 管 对 线程 的 支持 使 Java 语 言 的 

复杂 可 度 多 少 有 此 增加， 但 请 不 要 责怪 Java。 毕 竞 ， 和 用 线程 可 以 做 
益 的 事情 。 












































多 个 线程 可 能 共享 同一 个 资源 〈 比 如 一 个 对 象 里 的 内 存 ) ， 这 是 运用 线 
程 时 面临 的 最 大 的 一 个 抵 烦 。 必 须 保证 多 个 线程 不 会 同时 试图 读 取 和 修 
改 那 个 资源 。 这 要 求 技 巧 性 地 运用 synchronized (同步 ) 关键 字 。 它 是 
a 但 必须 真正 掌握 它 ， 因 为 假 徊 操作 不 当 ， 极 易 出 现 死 
:人 o 


除 此 以 外 ， 运 用 线程 时 还 要 注意 一 个 非常 特殊 的 问题 。 由 于 根据 Java 的 
设计 ， 它 允许 我 们 根据 需要 创建 任意 数量 的 线程 一 一 人 至少 理论 上 如 此 

〈 例 如， 假设 为 一 项 工程 方面 的 有 限 元 素 分 析 创 建 数 以 百 万 的 线程 ， 这 
对 Java 来 说 并 非 实际 ) 。 然 和 而， 我们 一 般 都 要 控制 自己 创建 的 线程 数量 
的 上 限 。 因 为 在 茶 些 情况 下 ， 大 量 线程 会 将 场面 变 得 一 团 糟 ， 所 以 工作 
都 会 几乎 陷于 停顿 。 临 界 点 并 不 象 对 象 那 样 可 以 达到 几 和 王 个 ， 而 是 在 

100 以 下 。 一 般 情 况 下 ， 我 们 只 创建 少数 几 个 关键 线程 ， 用 它们 解决 某 
个 特定 的 问题 。 这 时 数量 的 限制 问题 不 大 。 但 在 较 和 常规 的 一 些 设计 中 ， 

这 一 限制 确实 会 使 我 们 感到 束 手 束 脚 。 


大 家 要 注意 线程 处 理 中 一 个 不 是 十 分 直观 的 问题 。 由 于 采用 了 线程 < 调 
度 ” 机 制 ， 所 以 通过 在 run() 的 主 循环 中 插入 对 sleep0) 的 调用 ， 一 般 都 可 以 
使 自己 的 程序 运行 得 更 快 一 些 。 这 使 它 对 编程 技巧 的 要 求 非 党 高 ， 特 别 
是 在 更 长 的 延迟 似乎 反而 能 提高 性 能 的 时 候 。 当 然 ， 之 所 以 会 出 现 这 种 
情况 ， 是 由 于 在 正在 运行 的 线程 准备 进入 “休眠 ?状态 之 前 ， 较 短 的 延迟 
可 能 造成 “sleep0O 结 束 ” 调 度 机 制 的 中 断 。 这 便 强 迫 调 度 机 制 将 其 中 止 ， 
并 于 稍 后 重新 局 动 ， 以 便 它 能 做 完 自己 的 事情 ， 再 进入 休眠 状态 。 必 须 
多 想 一 想 ， 才 能 意识 到 事情 真正 的 麻烦 程度 。 


本 章 遗 漏 的 一 件 事情 是 一 个 动画 例子 ， 这 是 目前 程序 片 最 流行 的 一 种 应 
用 。 然 而 ，Java JDK 配 套 提供 了 解决 这 个 问题 的 一 上 整套 方案 (并 可 播放 
声音 ) ， 大 家 可 到 java.sun.com 的 演示 区 域 下 载 。 此 外 ， 我 们 完全 有 理 

由 相信 未 来 版 本 的 Java 会 提供 更 好 的 动画 支持 一 尽管 目前 的 Web 清 现 
出 了 与 传统 方式 完全 不 同 的 非 Java、 非 程序 化 的 许多 动画 方案 。 如 果 想 
系统 学 习 Java 动 男 的 工作 原理 ， 可 参考 《Core Java 核心 Java》 一 
书 ， 由 Cornell&Horstmann 编 竺 ，Prentice-Hall 于 1997 年 出 版 。 知 欲 更 深 

入 地 了 解 线程 处 理 ， 请 参考 《Concurrent Programming in Java——Java'} 
的 并 发 编程 》， 由 Doug Lea 编 著 ，Addison-Wiseley 于 1997 年 出 版 ， 或 者 
«Java Threads 一 一 Java 线 程 》，Oaks&Wong 编 车 ，O'Reilly 于 1997 年 出 





















































14.7 练习 


(1)” 从 Thread 继 承 一 个 类 ， 并 (过载) 履 羡 run(0) 方 法 。 在 run0 内 ， 打 印 
出 一 条 消息 ， 然 后 调用 sleepO0。 重 复 三 过 这 些 操作 ， 然 后 从 run0 返 回 。 
在 构建 器 中 放置 一 条 局 动 消 息 ， 并 上 履 盖 finalize0， 打 印 一 条 关闭 消息 。 
创建 一 个 独立 的 线程 类 ， 使 它 在 run0 内 调用 System.gcO0 和 
System.runFinalization()， 并 打印 一 条 消息 ， 表 明 调 用 成 功 。 创 建 这 两 种 
类 型 的 几 个 线程 ， 然 后 运行 它们 ， 看 看 会 发 生 什 么 。 


(2) 修改 Counter2.java， 使 线程 成 为 一 个 内 部 类 ， 而 且 不 需要 明确 保存 指 
向 Counter2 的 一 个 。 











(3) 修改 Sharing2.java， 在 TWoCounter 的 run0 方 法 内 部 添加 一 个 
synchronized (同步 ) 块 ， 而 不 是 同步 整个 run0) 方 法 。 


(4) “创建 两 个 Thread 子 类 ， 第 一 个 的 run0) 方 法 用 于 最 开始 的 启动 ， 并 捕 
获 第 二 个 Thread 对 象 的 句柄 ， 然 后 调用 wait0。 第 二 个 类 的 run0) 应 在 过 
几 秒 后 为 第 一 个 线程 调用 modifyAll()， 使 第 一 个 线程 能 打印 出 一 条 消 
息 。 


(5) 在 Ticker2 内 的 Counter5.java 中 ， 删 除 yield0， 并 解释 一 下 结果 。 用 一 
个 sleepO 换 掉 yield0， 再 解释 一 下 结果 。 


(6) 在 ThreadGroup1l.java 中 ， 将 对 sys.suspend0 的 调用 换 成 对 线程 组 的 一 
个 wait0 调 用 ， 令 其 等 候 2 秒 钟 。 为 了 保证 获得 正确 的 结果 ， 必 须 在 一 个 
同步 块 内 取得 sys 的 对 象 锁 。 


(7) 修改 Daemons.java， 使 main0 有 一 个 sleepO0， 而 不 是 一 个 readLine()。 
实验 不 同 的 睡眠 时 间 ， 看 看 会 有 什么 发 生 。 


(8) 到 第 7 章 〈 中 间 部 分 ) 找到 那个 GreenhouseControls.java 例 子 ， 它 应 该 
由 三 个 文件 构成 。 在 Event.java 中 ，Event 类 建立 在 对 时 间 的 监视 基础 

上 。 修 改 这 个 Event， 使 其 成 为 一 个 线程 。 然 后 修改 其 余 的 设计 ， 使 它 
们 能 与 新 的 、 以 线程 为 基础 的 Event 正 常 协作 。 














Pots we > 
P15% 网 络 编程 
历史 上 的 网 络 编程 都 倾 问 于 困难 、 复 杂 ， 而 且 极 易 出 错 。 


程序 员 必 须 掌 握 与 网 络 有 关 的 大 量 细 市 ， 有 时 甚至 要 对 硬件 有 深刻 的 认 
识 。 一 般 地 ， 我 们 需要 理解 连 网 协议 中 不 同 的 “ 层 ”(Layer) 。 而 且 对 

于 每 个 连 网 库 ， 一 般 都 包含 了 数量 众多 的 函数 ， 分 别 涉及 信息 块 的 连 

a erect wegen ome ree ee 
ERLE. 


但 是 ， 连 网 本 身 的 概念 并 不 是 很 难 。 我 们 想 获得 位 于 其 他 地 方 某 台 机 器 
上 的 信息 ， 并 把 它们 移 到 这 儿 ; 或 者 相反 。 这 与 读 写 文件 非常 相似 ， 只 
征文 件 存在 于 远程 机 器 上 ， 而 且 远程 机 器 有 权 决 定 如 何 处 理 我 们 请 求 或 
者 发 送 的 数据。 


Java 最 出 色 的 一 个 地 方 就 是 它 的 “无 痛苦 连 网 ?概念 。 有 关连 网 的 基层 细 
节 已 被 尽 可 能 地 提取 出 去 ， 并 隐藏 在 JVM 以 及 Java 的 本 机 安装 系统 里 进 
行 控制 。 我 们 使 用 的 编程 模型 是 一 个 文件 的 模型 ， 事 实 上 ， 网 络 连接 
〈 一 个 “ 套 接 字 ”) 已 被 封闭 到 系统 对 象 里 ， 所 以 可 象 对 其 他 数据 流 那 样 
采用 同样 的 方法 调用 。 除 此 以 外 ， 在 我 们 处 理 力 一 个 连 网 问题 一 一 同时 
EE S nec E 


本 章 将 用 一 系列 易 懂 的 例子 解释 Java 的 连 网 支持 。 




















15.1 机 器 的 标识 


当然 ， 为 了 分 辨 来自 别 处 的 一 台 机 器 ， 以 及 为 了 保证 目 己 连接 的 是 希望 
的 那 台 机 器 ， 必 须 有 一 种 机 制 能 独一无二 地 标识 出 网 络 内 的 每 台 机 器 。 
早期 网 络 只 解决 了 如 何在 本 地 网 络 环境 中 为 机 融 提 供 唯一 的 名 字 。 但 
Java 面 癌 的 是 整个 因特网 ， 这 要 求 用 一 种 机 制 对 来 自 世 界 各 地 的 机 需 进 
行 标识 。 为 达到 这 个 目的 ， 我 们 采用 了 IP〈 互 联网 地 址 ) 的 概念 。 卫 以 
两 种 形式 存在 着 : 


(1) 大 家 最 熟悉 的 DNS 〈 域 名 服务 ) 形式 。 我 自己 的 域名 是 
bruceeckel.com。 所 以 假定 我 在 自己 的 域内 有 一 台 名 为 Opus 的 计算 机 ， 
它 的 域名 就 可 以 是 Opus.bruceeckel.com。 这 正 是 大 家 问 其 他 人 发 送 电 子 
水 件 时 采用 的 名 字 ， 而 且 通 常 集成 到 一 个 万 维 网 (WWW) 地 址 里 。 


(2) 此 外 ， 亦 可 采用 “四 点 ”格式 ， 亦 即 由 点 号 ©.) 分 隔 的 四 组 数字 ， 比 
如 202.98.32.111。 


不 管 哪 种 情况 ， 卫 地 址 在 内 部 都 表达 成 一 个 由 32 个 二 进 制 位 Chit) 构成 
WAF GERO) ， 所 以 IP 地 址 的 每 一 组 数字 都 不 能 超过 255。 利 用 由 

java.net 提 供 的 static ”InetAddress.getByName()， 我 们 可 以 让 一 个 特定 的 
Java 对 象 表 达 上 述 任 何 一 种 形式 的 数字 。 结 果 是 类 型 为 InetAddress 的 一 
由 




















O: 这 意味 着 最 多 只 能 得 到 40 亿 左右 的 数字 组 合 ， 全 世界 的 人 很 快 就 会 
把 它 用 光 。 但 根据 目前 正在 研究 的 新 IP 编 址 方案 ， 它 将 采用 128 bit 的 数 
字 ， 这 样 得 到 的 唯一 性 人 P 地 址 也 许 在 儿 百 年 的 时 间 里 都 不 会 用 完 。 


作为 运用 InetAddress.getByName() 一 个 简单 的 例子 ， 请 考虑 假设 自己 有 
一 家 拨号 连接 因特网 服务 提供 者 USP) ， 那 么 会 发 生 什 么 情况 。 每 次 
拨号 连接 的 时 候 ， 都 会 分 配 得 到 一 个 临时 卫 地 址 。 但 在 连接 期 间 ， 那 个 
IP 地 址 拥有 与 因特网 上 其 他 IP 地 址 一 样 的 有 效 性 。 如 果 有 人 按照 你 的 人 P 
地 址 连接 你 的 机 器 ， 他 们 就 有 可 能 使 用 在 你 机 器 上 运行 的 Web 或 者 FTP 
服务 器 程序 。 当 然 这 有 个 前 提 ， 对 方 必须 准确 地 知道 你 目前 分 配 到 的 
IP。 由 于 每 次 拨号 连接 获得 的 IP 都 是 随机 的 ， 怎 样 才能 准确 地 掌握 你 的 
IP 呢 ? 














下 面 这 个 程序 利用 InetAddress.getByName0) 来 产生 你 的 耳 地 址 。 为 了 让 
它 运 行 起 来 ， 事 先 必 须知 道 计 算 机 的 名 字 。 访 程序 只 在 Windows 95 中 进 
行 了 测试 ， 但 大 家 可 以 依次 进入 自己 的 “开始 "”、“ 设 置 "、“ 控 制 面 

板 ?、“ 网 络 ”"”， 然 后 进入 “标识 ”卡片 。 其 中 ,，“ 计 算 机 名 称 ” 就 是 应 在 命令 
行 输 入 的 内 容 。 


//: WhoAmI .java 








// Finds out your network address when you're 
// connected to the Internet. 
package c15; 
import java.net.*; 
public class WhoAmI { 
public static void main(String[] args) 
throws Exception { 
if(args.length != 1) { 
System.err.printin( 
"Usage: WhoAmI MachineName"); 
System.exit(1); 
} 
InetAddress a = 
InetAddress.getByName(args[0]); 


System.out.println(a); 


SP 
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人 ”的 意思 。 我 在 这 台 机 器 上 有 一 个 很 大 的 硬盘 〉 。 所 以 一 旦 连 通 我 的 
ISP， 惑 象 下 面 这 样 执行 程序 : 


java whoAmI Colossus 
43 Bl) AGRA PI NE CA, BL RE RABE HAD : 
Colossus/202.98.41.151 


假如 我 把 这 个 地 址 告诉 一 位 朋友 ， 他 就 可 以 立即 登录 到 我 的 个 人 Web 服 
务 堪 ， 只 需 指定 目标 地 址 http:/202.98.41.151 即 可 《当然 ， 我 此 时 不 能 断 
线 ) 。 有 些 时 候 ， 这 是 向 其 他 人 发 送信 息 或 者 在 自己 的 Web 站 点 正式 出 
台 以 前 进行 测试 的 一 种 方便 手段 。 


15.1.1 服务 器 和 客户 机 


网 络 最 基本 的 精神 就 是 让 两 台 机 强 连 接 到 一 起 ， 并 相互 “交谈 ”或 者 “ 沟 
通 "。 一 旦 两 合 机 器 部 发 现 了 对 方 ， 就 可 以 展开 一 次 令 人 愉快 的 双 同 对 
话 。 但 它们 怎样 才能 “发 现 ?对 方 呢 ? 这 就 象 在 游乐 园 里 那样 : 一 合 机 顺 
不 得 不 停留 在 个 地 方 ， 侦 听 其 他 机 器 说 :“ 嘿 ， 你 在 哪里 呢 ? 


“停留 在 一 个 地 方 ”* 的 机 器 叫 作 “服务 器 ”CServer) ; 到处“ 找 人 ”的 机 器 
则 叫 作 “客户 机 ”(〈Client) 或 者 “客户 ”。 它们 之 间 的 区 别 喉 4 有 在 客户 机 
试图 同 服务 器 连接 的 时 候 才 显得 非常 明显 。 - 且 连 通 ， 就 变 成 了 一 种 双 
癌 通 信 ， 谁 来 扮演 服务 器 或 者 客户 机 便 显 得 不 那么 重要 了 。 


所 以 服务 器 的 主要 任务 是 侦 听 建立 连接 的 请 求 ， 这 是 由 我 们 创建 的 特定 
服务 器 对 象 完 成 的 。 而 客户 机 的 任务 是 试 着 与 一 台 服 务 器 建立 连接 ， 这 
是 由 我 们 创建 的 特定 客户 机 对 象 完成 的 。 一 旦 连接 建 好 ， 那 么 无 论 在 服 
务 右 端 还 是 客户 机 端 ， 连 接 只 是 麻 术 般 地 变 成 了 一 个 I0 数 据 流 对 象 。 从 
这 时 开始 ， 我 们 可 以 象 读 写 一 个 普通 的 文件 那样 对 待 连接 。 所 以 一 旦 建 
好 连接 ， 我 们 只 需 象 第 10 章 那样 使 用 自己 熟悉 的 IO 命 令 即 可 。 这 正 是 

Java 连 网 最 方便 的 一 个 地 方 。 


1. 在 没有 网 络 的 前 提 下 测试 程序 
由 于 多 种 潜在 的 原因 ， 我 们 可 能 没有 一 台 客 户 机 、 服 务 器 以 及 一 个 网 络 
































来 测试 目 己 做 好 的 程序 。 我 们 也 许 是 在 一 个 课堂 环境 中 进行 练习 ， 或 者 
写 出 的 是 一 个 不 十 分 可 靠 的 网 络 应 用 ， 还 能 拿 到 网 络 上 去 。IP 的 设计 者 
注意 到 了 这 个 问题 ， 并 建立 了 一 个 特殊 的 地 址 localhost 来 满足 
非 网 络 环境 中 的 测试 要 求 。 在 Java 中 产生 这 个 地 址 最 一 般 的 做 法 是 : 








InetAddress addr = InetAddress.getByName(null); 


如 果 向 getByName( 传 递 一 个 null (F) 值 ， 就 默认 为 使 用 localhost。 我 
们 用 InetAddress 对 特定 的 机 器 进行 索引 ， 而 且 必 须 在 进行 进一步 的 操作 
之 前 得 到 这 个 ImetAddress《〈 互 联网 地 址 ) 。 我 们 不 可 以 操纵 一 个 
InetAddress 的 内 容 《〈 但 可 把 它 打 印 出 来 ， 就 象 下 一 个 例子 要 演示 的 那 
样 ) 。 创 建 InetAddress 的 唯一 途径 就 是 那个 类 的 static《〈 静 态 ) 成 员 方 法 
getByName() 《〈 这 是 最 常用 的 ) 、getAllByName() 或 者 getLocalHost()。 


为 得 到 本 地 主机 地 址 ， 亦 可 癌 其 直接 传递 字 串 "localhost": 








InetAddress.getByName("localhost"); 

或 者 使 用 它 的 保留 IP 地 址 (四 点 形式 ) ， 就 象 下 面 这 样 : 
InetAddress.getByName("127.0.0.1"); 

这 三 种 方法 得 到 的 结果 是 一 样 的 。 

15.1.2 山口 ， 机 堪 内 独一无二 的 场所 


有 些 时 候 ， 一 个 耳 地 址 并 不 足以 完整 标识 一 个 服务 器 。 这 和 是 由 于 在 一 人 台 
物理 性 的 机 器 中 ， 往 往 运 行 着 多 个 服务 器 〈 程 序 ) 。 由 也 表达 的 每 台 机 
ata Sim” (Port) 。 我 们 设置 一 个 客户 机 或 者 服务 器 的 时 候 ， 
必须 选择 一 个 无 论 客 户 机 还 是 服务 器 都 认可 连接 的 端口 。 惑 象 我 们 去 拜 
会 某 人 时 ，IP 地 址 是 他 居住 的 房子 ， 而 端口 是 他 在 的 那个 房间 。 


注意 端口 并 不 是 机 器 上 一 个 物理 上 存在 的 场所 ， 而 是 一 种 软件 抽象 〈 主 
要 是 为 了 表述 的 方便 ) 。 客 户 程序 知道 如 何 通 过 机 堪 的 耳 地 址 同 它 连 
接 ， 但 怎样 才能 同上 自己 真正 需要 的 那 种 服务 连接 呢 (一般 每 个 端口 都 运 
行 着 一 种 服务 ， 一 合 机 器 可 能 提供 了 多 种 服务 ， 比 如 HITP 和 FTP 等 
等 ) ? 端口 编号 在 这 里 扮演 了 重要 的 角色 ， 它 是 必需 的 一 种 二 级 定 址 措 
施 。 也 就 是 说 ， 我 们 请 求 一 个 特定 的 端口 ， 便 相当 于 请 求 与 那个 端口 编 








号 关联 的 服务 。“ 报 时 ” 便 是 服务 的 一 个 典型 例子 。 通 常 ， 每 个 服务 部 同 
一 台 特 定 服 务 器 机 器 上 的 一 个 独一无二 的 器 口 编号 关联 在 一 起 。 客 户 程 
序 必须 事先 知道 目 己 要 求 的 那 项 服务 的 运行 端口 号 。 


系统 服务 保留 了 使 用 端口 1 到 端口 1024 的 权力 ， 所 以 不 应 让 自己 设计 的 
服务 占用 这 些 以 及 其 他 任何 已 知 正在 使 用 的 端口 。 本 书 的 第 一 个 例子 将 
使 用 端口 8080 (为 追忆 我 的 第 一 台 机 器 使 用 的 老式 8 位 Intel 8080 心 片 ， 
那 是 一 部 使 用 CP/M 操 作 系 统 的 机 子 ) 。 








15.2 套 接 字 


“ 套 接 字 ”或 者 “插座 ”(Socket) 也 是 一 种 软件 形式 的 抽象 ， 用 于 表达 两 
台 机 器 间 一 个 连接 的 “终端 ”。 针 对 一 个 特定 的 连接 ， 每 侣 机 器 上 都 有 一 
个 “ 套 接 字 ”， 可 以 想象 它们 之 间 有 一 条 虚拟 的 “ 线 线 ”。 线 线 的 每 一 端 都 
插入 一 个 “ 套 接 字 ? 或 者 “插座 里。 当然 ， 机 器 之 间 的 物理 性 硬件 以 及 电 
BSE EAA 的 。 抽 象 的 基本 宗旨 是 让 我 们 尽 可 能 不 必 知 道 那些 
ANA 


在 Java 中 ， 我 们 创建 一 个 套 接 字 ， 用 它 建立 与 其 他 机 器 的 连接 。 从 套 接 
字 得 到 的 结果 是 一 个 ImputStream 以 及 OutputStream 〈 知 使 用 恰当 的 转换 
器 ， 则 分 别 是 Reader 和 Writer) ， 以 便 将 连接 作为 一 个 10 流 对 象 对 待 。 
有 两 个 基于 数据 流 的 套 接 字 类 : ServerSocket， 服 务 器 用 它 “ 侦 听 ” 进 入 
的 连接 ; 以 及 Socket， 客 户 用 它 初 始 一 次 连接 。 一 旦 客户 (程序 ) 申请 
建立 一 个 套 接 字 连接 ，ServerSocket 就 会 返回 〈 通 过 accept0 方 法 ) 一 个 
对 应 的 服务 咒 端 套 接 字 ， 以 便 进 行 直接 通信 。 从 此 时 起 ， 我 们 就 得 到 了 
真正 的 “ 套 接 字 一 套 接 字 ” 连 接 ， 可 以 用 同样 的 方式 对 待 连接 的 两 端 ， 
为 它们 本 来 就 是 相同 的 ! 此 时 可 以 利用 getInputStream( 以 及 
getOutputStream() 从 每 个 套 接 字 产生 对 应 的 InputStream 和 OutputStream 对 
象 。 这 些 数据 流 必 须 封 装 到 绥 冲 区 内 。 可 按 第 10 章 介绍 的 方法 对 类 进行 
格式 化 ， 就 象 对 竺 其 他 任何 流 对 象 那样 。 


对 于 Java 库 的 命名 机 制 ，ServerSocket 〈 服 务 器 套 接 字 ) 的 使 用 无 疑 是 容 
易 产 生 混 消 的 又 一 个 例证 。 大 家 可 能 认为 ServerSocket 最 好 叫 

作 “ServerConnector”( 服 务 器 连接 器 ) ， 或 者 其 他 什么 名 字 ， 只 是 不 要 
在 其 中 安插 一 个 “Socket”。 也 可 能 以 为 ServerSocket 和 Socket 都 应 从 一 些 
通用 的 基础 类 继承 。 事 实 上 ， 这 两 种 类 确实 包含 了 几 个 通用 的 方法 ， 但 
还 不 够 资格 把 它们 赋 给 一 个 通用 的 基础 类 。 相 反 ，ServerSocket 的 主要 
任务 是 在 那里 耐心 地 等 候 其 他 机 器 同 它 连接 ， 再 返回 一 个 实际 的 
Socket。 这 正 是 “ServerSocket” 这 个 命名 不 恰当 的 地 方 ， 因 为 它 的 目标 不 
eee 而 是 在 其 他 人 同 它 连接 的 时 候 产 生 一 个 Socket 
对 象 。 


然而 ，ServerSocket 确 实 会 在 主机 上 创建 一 个 物理 性 的 “服务 器 ?或 者 侦 
听 用 的 套 接 字 。 这 个 套 接 字 会 侦 听 进入 的 连接 ， 然 后 利用 accept0 方 法 
返回 一 个 “已 建立 ” 套 接 字 ( 本 地 和 远程 端点 均 已 定义 ) 。 容 易 混 请 的 地 





























方 是 这 两 个 套 接 字 〈 侦 听 和 已 建立 ) 都 与 相同 的 服务 器 套 接 字 关 联 在 一 
起 。 侦 听 套 接 字 只 能 接收 新 的 连接 请 求 ， 不 能 接收 实际 的 数据 包 。 所 以 
尽管 ServerSocket 对 于 编程 并 无 太 大 的 意义 ， 但 它 确 实 是 “物理 性 ”的 。 


创建 一 个 ServerSocket 时 ， 只 和 需 为 其 赋予 一 个 端口 编写。 不必 把 一 个 IP 
地 址 分 配 它 ， 因 为 它 已 经 在 自己 代表 的 那 台 机 器 上 了 。 但 在 创建 一 个 
Socket 时 ， 却 必须 同时 赋予 IP 地 址 以 及 要 连接 的 端口 编号 〈 另 一 方面 ， 
从 ServerSocket'.acceptO 返 回 的 Socket 已 经 包含 了 所 有 这 些 信息 ) 。 


15.2.1 一 个 简单 的 服务 器 和 客户 机 程序 


这 个 例子 将 以 最 简单 的 方式 运用 套 接 字 对 服务 器 和 客户 机 进行 操作 。 服 
务 器 的 全 部 工作 就 是 等 候 建立 一 个 连接 ， 然 后 用 那个 连接 产生 的 Socket 
创建 一 个 mputStream 以 及 一 个 OutputStream。 在 这 之 后 ， 它 从 
InputStream 读 入 的 所 有 东西 都 会 反馈 给 OutputStream， 直 到 接收 到 行 中 
止 CEND) 为 止 ， 最 后 关闭 连接 。 


客户 机 连接 与 服务 器 的 连接 ， 然 后 创建 一 个 OutputStream。 文 本 行 通过 
OutputStream 发 送 。 客 户 机 也 会 创建 一 个 InputStream， 用 它 收听 服务 器 
说 些 什 么 《〈 本 例 只 不 过 是 反馈 回来 的 同样 的 字 名 ) 。 


服务 器 与 客户 机 《程序 ) 都 使 用 同样 的 端口 号 ， 而 且 客 户 机 利用 本 地 主 
机 地 址 连接 位 于 同一 合 机 器 中 的 服务 器 《程序 ) ， 所 以 不 必 在 一 个 物理 
性 的 网 络 里 完成 测试 《在 荣 些 配置 环境 中 ， 可 能 需要 同 真 正 的 网 络 建立 
连接 ， 人 否则 程序 不 能 工作 一 一 尽管 实际 并 不 通过 那个 网 络 通信 ) 。 


下 面 是 服务 器 程序 : 


//: JabberServer.java 

















// Nery simple server that just 

// echoes whatever the client sends. 
import java.io.*; 

import java.net.*; 


public class JabberServer { 


// Choose a port outside of the range 1-1024: 
public static final int PORT = 8080; 
public static void main(String[] args) 
throws IOException { 
ServerSocket s = new ServerSocket(PORT); 
System.out.println("Started: " + s); 
try { 
// Blocks until a connection occurs: 
Socket socket = s.accept(); 
try { 
System.out.printin( 
"Connection accepted: "+ socket); 
BufferedReader in = 
new BufferedReader ( 
new InputStreamReader ( 
socket.getInputStream())); 
// Output is automatically flushed 
// by PrintWwriter: 
PrintWriter out = 
new Printwriter ( 
new Bufferedwriter ( 
new OutputStreamwriter ( 


socket. getOutputStream())),true); 


while (true) { 
String str = in.readLine(); 
if (str.equals("END")) break; 
System.out.println("Echoing: " + str); 
out.println(str); 
} 
// Always close the two sockets... 
} finally { 
System.out.println("closing..."); 
socket.close(); 
} 
} finally { 


s.close(); 


} 
ke 


可 以 看 到 ，ServerSocket 需 要 的 只 是 一 个 端口 编写， 不 需要 IP 地 址 〈( 因 
为 它 就 在 这 人 台 机 器 上 运行 ) 。 调 用 acceptO0 时 ， 方 法 会 暂时 陷入 停顿 状 
态 〈 堵 塞 ) ， 直 到 某 个 客户 党 试 同 它 建 立 连接 。 换 言 之 ， 尽 管 它 在 那里 
等 候 连接 ， 但 其 他 进程 仍 能 正常 运行 〈 参 考 第 14 章 ) 。 建 好 一 个 连接 以 
后 ，accept0 就 会 返回 一 个 Socket 对 象 ， 它 是 那个 连接 的 代表 。 


清除 套 接 字 的 责任 在 这 里 得 到 了 很 艺术 的 处 理 。 假 如 ServerSocket 构 建 
器 失败 ， 则 程序 简单 地 退出 (注意 必须 保证 ServerSocket 的 构建 器 在 失 
败 之 后 不 会 留 下 任何 打开 的 网 络 套 接 字 ) 。 人 针对 这 种 情况 ，main() 
会 “ 搓 ” 出 一 个 IOException 违 例 ， 所 以 不 必 使 用 一 个 try 块 。 知 








ServerSocket 构 建 器 成 功 执行 ， 则 其 他 所 有 方法 调用 都 必须 到 一 个 try- 
finally 代 码 块 里 寻求 保护 ， 以 确保 无 论 块 以 什么 方式 留 下 ，ServerSocket 
都 能 正确 地 关闭 。 


同样 的 道理 也 适用 于 由 acceptO 返 回 的 Socket。 若 acceptO 失 败 ， 那 么 我 们 
必须 保证 Socket 不 再 存在 或 者 含有 任何 资源 ， 以 便 不 必 清 除 它 们 。 但 假 
知 执行 成 功 ， 则 后 续 的 语句 必须 进入 一 个 try-finally 块 内 ， 以 保障 在 它们 
失败 的 情况 下 ，Socket 仍 能 得 到 正确 的 清除 。 由 于 套 接 字 使 用 了 重要 的 
非 内 存 资源 ， 所 以 在 这 里 必须 特别 谨 懂 ， 必 须 上 自己 动手 将 它们 清除 
(Java 中 没有 提供 “破坏 器 ”来 帮助 我 们 做 这 件 事 情 〉。 


无 论 ServerSocket 还 是 由 accept0 产 生 的 Socket 都 打印 到 System.out 里 。 这 
意味 着 它们 的 toString 方 法 会 得 到 自动 调用 。 这 样 便 产 生 了 : 


ServerSocket[addr=0.0.0.0, PORT=0, Llocalport=8080 ] 


Socket [addr=127.0.0.1, PORT=1077, localport=80860 | 





大 家 不 久 束 会 看 到 它们 如 何 与 客户 程序 做 的 事情 配合 。 


程序 的 下 一 部 分 看 来 似乎 仅仅 是 打开 文件 ， 以 便 读 取 和 写 入 ， 只 是 
InputStream 和 OutputStream 是 从 Socket 对 象 创建 的 。 利 用 两 个 “转换 器 ”类 
InputStreamReader 和 OutputStreamWriter，InputStream 和 OutputStream 对 
象 已 经 分 别 转换 成 为 Java ”1.1 的 Reader 和 Writer 对 象 。 也 可 以 直接 使 用 
Javal.0 的 PmputStream 和 OutputStream 类 ， 但 对 输出 来 说 ， 使 用 Writer 方 式 
具有 明显 的 优势 。 这 一 优势 是 通过 PrintWriter 表 现 出 来 的 ， 它 有 一 个 过 
载 的 构建 问 ， 能 获取 第 二 个 参数 一 个 布尔 值 标 志 ， 指 问 是 否 在 每 一 
次 println(0) 结 束 的 时 候 自 动 刷 新 输出 (但 不 适用 于 print0 语 句 ) 。 每 次 写 
入 了 输出 内 容 后 ( 写 进 out〉， 它 的 缓冲 区 必须 刷新 ， 使 信息 能 正式 通 
过 网 络 传递 出 去 。 对 目前 这 个 例子 来 说 ， 刷 新 显得 尤为 重要 ， 因 为 客户 
和 服务 器 在 采取 下 一 步 操作 之 前 都 要 等 待 一 行文 本 内 容 的 到 达 。 知 刷新 
没有 发 生 ， 那 么 信息 不 会 进入 网 络 ， 除 非 缓冲 区 满 ( 洲 出 ) ， 这 会 为 本 
例 带 来 许多 问题 。 


编写 网 络 应 用 程序 时 ， 需 要 特别 注意 自动 刷新 机 制 的 使 用 。 每 次 刷新 组 
冲 区 时 ， 必 须 创 建 和 发 出 一 个 数据 包 《〈 数 据 封 ) 。 就 目前 的 情况 来 说 ， 











这 正 是 我 们 所 希望 的 ， 因 为 假如 包 内 包含 了 还 没有 发 出 的 文本 行 ， 服 务 
器 和 客户 机 之 间 的 相互 “握手 ?就 会 停止 。 换 句 话 说， 一 行 的 末尾 就 是 一 
条 消息 的 来 尾 。 但 在 其 他 许多 情况 下 ， 消 妃 并 不 是 用 行 分 隔 的 ， 所 以 不 
如 不 用 目 动 刷新 机 制 ， 而 用 内 建 的 缓冲 区 判决 机 制 来 决定 何 时 发 送 一 个 
oe 这 样 一 来 ， 我 们 可 以 发 出 较 大 的 数据 包 ， 而 且 处 理 进 程 也 能 加 








注意 和 我 们 打开 的 几乎 所 有 数据 流 一 样 ， 它 们 都 要 进行 缓冲 处 理 。 本 章 
末尾 有 一 个 练习 ， 清 楚 展 现 了 假如 我 们 不 对 数据 流 进行 缓冲 ， 那 么 会 得 
到 什么 样 的 后 末 《〈 速 度 会 变 慢 ) 。 


无 限 while 循 环 从 BufferedReader in 内 读 取 文本 行 ， 并 将 信息 写 入 
System.out， 然 后 写 入 PrintWriter.out。 注 意 这 可 以 是 任何 数据 流 ， 它 们 
只 是 在 表面 上 同 网 络 连接 。 


客户 程序 发 出 包含 了 "END" 的 行 后 ， 程 序 会 中 止 循环 ， 并 关闭 Socket。 
下 面 是 客户 程序 的 源码 : 


//: JabberClient.java 


// Very simple client that just sends 
// lines to the server and reads lines 
// that the server sends. 
import java.net.*; 
import java.io.*; 
public class JabberClient { 
public static void main(String[] args) 
throws IOException { 
// Passing null to getByName() produces the 


// special "Local Loopback" IP address, for 


// testing on one machine w/o a network: 
InetAddress addr = 
InetAddress.getByName(null); 
// Alternatively, you can use 
// the address or name: 
// InetAddress addr = 
// InetAddress.getByName("127.0.0.1"); 
// InetAddress addr = 
// InetAddress.getByName("localhost"); 
System.out.printin("addr = " + addr); 
Socket socket = 
new Socket(addr, JabberServer.PORT); 
// Guard everything in a try-finally to make 
// sure that the socket is closed: 
try { 
System.out.println("socket = " + socket); 
BufferedReader in = 
new BufferedReader ( 
new InputStreamReader ( 
socket.getInputStream())); 
// Output is automatically flushed 
// by Printwriter: 


PrintWriter out = 


new Printwriter ( 
new Bufferedwriter ( 
new OutputStreamwr iter ( 
socket.getOutputStream())), true); 
for(int i = 0; i < 10; i ++) { 
out.printin("howdy " + i); 
String str = in.readLine(); 
System.out.println(str); 
} 
out.printin("END"); 
} finally { 
System.out.println("closing..."); 


socket.close(); 


} 
ke 





在 main() 中 ， 大 家 可 看 到 获得 本 地 主机 IP 地 址 的 InetAddress 的 三 种 途 

径 : 使 用 null， 使 用 localhost， 或 者 直接 使 用 保留 地 址 127.0.0.1。 当 然 ， 
如 果 想 通过 网 络 同一 台 远 程 主机 连接 ， 也 可 以 换 用 那 台 机 器 的 了 地 址 。 
打印 出 InetAddress _ addr 后 〈 通 过 对 toString(0) 方 法 的 自动 调用 ) ， 结 果 如 
T 


localhost/127.0.0.1 


通过 向 getByName() 传 递 一 个 null， 它 会 默认 寻找 localhost， 并 生成 特殊 
的 保留 地 址 127.0.0.1。 注 意 在 名 为 socket 的 套 接 字 创建 时 ， 同 时 使 用 了 


InetAddress 以 及 端口 号 。 打 印 这 样 的 某 个 Socket 对 象 时 ， 为 了 真正 理解 
它 的 含义 ， 请 记 住 一 次 独一无二 的 因特网 连接 是 用 下 述 四 种 数据 标识 
HJ: clientHost (XPF EHL) ~ clientPortNumber (客户 端口 号 ) 、 
serverHost (服务 主机 〉 以 及 serverPortNumber (服务 端口 号 ) 。 服 务 程 
序 启动 后 ， 会 在 本 地 主机 〈127.0.0.1) 上 建立 为 它 分 配 的 端口 

(8080) 。 一 且 客 户 程序 发 出 请 求 ， 机 器 上 下 一 个 可 用 的 端口 就 会 分 配 
给 它 《〈 这 种 情况 下 是 1077) ， 这 一 行动 也 在 与 服务 程序 相同 的 机 右 
(127.0.0.1) 上 进行 。 现 在 ， 为 了 使 数据 能 在 客户 及 服务 程序 之 间 来 回 
传送 ， 每 一 问 都 需要 知道 把 数据 发 到 哪里 。 所 以 在 同一 个 “已 知 ?服务 程 
序 连 接 的 时 候 ， 客 户 会 发 出 一 个 “返回 地 址 ”， 使 服务 器 程序 知道 将 自己 
的 数据 发 到 哪儿 。 我 们 在 服务 器 端的 示范 输出 中 可 以 体会 到 这 一 情况 : 


Socket[addr=127.0.0.1,port=1077,localport=8080] 


这 意味 着 服务 器 刚才 已 接受 了 来 自 127.0.0.1 这 台 机 器 的 端口 1077 的 连 
接 ， 同 时 监听 自己 的 本 地 端口 (8080) 。 而 在 客户 端 : 


Socket[Laddr=localhost/127.0.0.1,PORT=8080,localport=1077] 


这 意味 着 客户 已 用 自己 的 本 地 端口 1077 与 127.0.0.1 机 器 上 的 端口 8080 建 
WS 连接 。 


大 家 会 注意 到 每 次 重新 月 动 客户 程序 的 时 候 ， 本 地 端口 的 编号 都 会 增 
加 。 这 个 编号 从 1025《 刚 好 在 系统 保留 的 1-1024 之 外 ) 开始 ， 并 会 一 直 
增加 下 去 ， 除 非 我 们 重启 机 器 。 寿 重新 局 动机 器 ， 端 口号 仍然 会 从 1025 
开始 增值 (在 Unix 机 器 中 ， 一 旦 超过 保留 的 套 按 字 范围 ， 数 字 就 会 再 次 
从 最 小 的 可 用 数字 开始 ) 。 


创建 好 Socket 对 象 后 ， 将 其 转换 成 BufferedReader 和 PrintWriter 的 过 程 便 
与 在 服务 器 中 相同 (同样 地 ， 两 种 情况 下 都 要 从 一 个 Socket 开 始 )。 在 
这 里 ， 客 户 通过 发 出 字 串 "howdy"， 并 在 后 面 跟随 一 个 数字 ， 从 而 初始 
化 通信 。 注 意 缓冲 区 必须 再 次 刷新 (这 是 自动 友 生 的 ， 通 过 传递 给 
PrintWriter 构 建 器 的 第 二 个 参数 ) 。 知 缓冲 区 没有 刷新 ， 那 么 整个 会 话 
(通信 ) 都 会 被 挂 起 ， 因 为 用 于 初始 化 的 howdy” 永 远 不 会 发 送出 去 
《缓冲 区 不 够 满 ， 不 足以 造成 发 送 动作 的 上 自动 进行 ) 。 从 服务 器 返回 的 
每 一 行 都 会 写 入 System.out， 以 验证 一 切 都 在 正常 运转 。 为 中 止 会 话 ， 
二 
违例 。 





























大 家 在 这 里 可 以 看 到 我 们 采用 了 同样 的 措施 来 确保 由 Socket 代 表 的 网 络 
资源 得 到 正确 的 清除 ， 这 是 用 一 个 try-finally 块 实现 的 。 


套 接 字 建立 了 一 个 “专用 ?连接 ， 它 会 一 直 持续 到 明确 断 开 连接 为 止 〈 专 
用 连接 也 可 能 间接 性 地 断 开 ， 前 提 是 茶 一 器 或 者 中 间 的 某 条 链 路 出 现 故 
RAM Hit) 。 这 意味 着 参与 连接 的 双方 都 被 锁定 在 通信 中 ， 而 且 无 论 是 
含有 效 据 传递 ， 连 接 都 会 连续 处 于 开放 状态 。 从 表面 看 ， 这 似乎 是 一 种 
合理 的 连 网 方式 。 然 而 ， 它 也 为 网 络 带 来 了 额外 的 开销 。 本 章 后 面 会 介 
绍 进行 连 网 的 态 一 种 方式 。 采 用 那 种 方式 ， 连 接 的 建立 只 是 暂时 的 。 








15.3 服务 多 个 客户 


JabberServer 可 以 正常 工作 ， 但 每 次 只 能 为 一 个 客户 程序 提供 服务 。 在 

典型 的 服务 器 中 ， 我 们 希望 同时 能 处 理 多 个 客户 的 请 求 。 解 决 这 个 问题 
的 关键 就 是 多 线程 处 理 机 制 。 而 对 于 那些 本 和 丑 不 支持 多 线程 的 语言 ， 达 
到 这 个 要 求 无 疑 是 异常 困难 的 。 通 过 第 14 间 的 学 习 ， 大 家 已 经 知道 Java 
己 对 多 线程 的 处 理 进行 了 尽 可 能 的 简化 。 由 于 Java 的 线程 处 理 方式 非常 
直接 ， 所 以 让 服务 器 控制 多 名 客户 并 不 是 件 难事 。 


最 基本 的 方法 是 在 服务 器 (程序 ) 里 创建 单个 ServerSocket， 并 调用 
accept() 来 等 候 一 个 新 连接 。 一 旦 accept0 返 回 ， 我 们 就 取得 结果 获得 的 
Socket， 并 用 它 新 建 一 个 线程 ， 令 其 只 为 那个 特定 的 客户 服务 。 然 后 再 
调用 acceptO0， 等 候 下 一 次 新 的 连接 请 求 。 

对 于 下 面 这 段 服 务 器 代码 ， 大 家 可 发 现 它 与 JabberServer.java 例 子 非常 相 
似 ， 只 是 为 一 个 特定 的 客户 提供 服务 的 所 有 操作 都 已 移入 一 个 独立 的 线 


程 类 中 ， 


//: MultiJabberServer.java 

















// A server that uses multithreading to handle 
// any number of clients. 
import java.io.*; 
import java.net.*; 
class ServeOneJabber extends Thread { 
private Socket socket; 
private BufferedReader in; 
private PrintwWriter out; 
public ServeOneJabber(Socket s) 


throws IOException { 


socket = s; 
in = 
new BufferedReader ( 
new InputStreamReader ( 
socket.getInputStream())); 
// Enable auto-flush: 
out = 
new Printwriter ( 
new Bufferedwriter ( 
new OutputStreamwr iter ( 
socket.getOutputStream())), true); 
// If any of the above calls throw an 
// exception, the caller is responsible for 
// closing the socket. Otherwise the thread 
// will close it. 
start(); // Calls run() 


} 


public void run() { 
try { 
while (true) { 
String str = in.readLine(); 
if (str.equals("END")) break; 


System.out.printin("Echoing: " + str); 


out.println(str); 
} 
System.out.println("closing..."); 
} catch (IOException e) { 
} finally { 
try { 
socket.close(); 


} catch(I0Exception e) {} 


J 


public class MultiJabberServer { 
static final int PORT = 8080; 
public static void main(String[] args) 
throws IOException { 
ServerSocket s = new ServerSocket(PORT); 
System.out.printin("Server Started"); 
try { 
while(true) { 
// Blocks until a connection occurs: 
Socket socket = s.accept(); 


try { 


new ServeOneJabber (socket); 


} catch(IOException e) { 
// If it fails, close the socket, 
// otherwise the thread will close it: 


socket.close(); 


} 
} finally { 


s.close(); 


} 
} ///:~ 


每 次 有 新 客户 请 求 建立 一 个 连接 时 ，ServeOneJabber 线 程 都 会 取得 由 
accept() 在 main() 中 生成 的 Socket 对 象 。 然 后 和 往常 一 样 ， 它 创建 一 个 
BufferedReader， 并 用 Socket 自 动 刷新 PrintWriter 对 象 。 最 后 ， 它 调用 
Thread 的 特殊 方法 start()， 令 其 进行 线程 的 初始 化 ， 然 后 调用 run()。 这 
里 采取 的 操作 与 前 例 是 一 样 的 : 从 套 扫 字 读 入 某 些 东 西 ， 然 后 把 它 原样 
反馈 回去 ， 直 到 过 到 一 个 特殊 的 "END" 结 束 标志 为 止 。 


同样 地 ， 套 接 字 的 清除 必须 进行 谨慎 的 设计 。 束 目前 这 种 情况 来 说 ， 套 
接 字 是 在 ServeOneJabber 外 部 创建 的 ， 所 以 清除 工作 可 以 “共享 >”。 知 
ServeOneJabber 构 建 问 失败， 那么 只 需 回 调用 者 “ 掷 ?出 一 个 违例 即 可 ， 
然后 由 调用 者 负责 线程 的 清除 。 但 假如 构建 嚣 成功， 那么 必须 由 
ServeOneJabber 对 象 负 责 线 程 的 清除 ， 这 是 在 它 的 runO 里 进行 的 。 


请 注意 MultiJabberServer 有 多 么 简单 。 和 以 前 一 样 ， 我 们 创建 一 个 
ServerSocket， 并 调用 accept() 允 许 一 个 新 连接 的 建立 。 但 这 一 次 ， 
accept() 的 返回 值 ( 一 个 套 接 字 ) 将 传递 给 用 于 ServeOneJabber 的 构建 
器 ， 由 它 创 建 一 个 新 线程 ， 并 对 那个 连接 进行 控制 。 连 接 中 断后 ， 线 程 
便 可 简单 地 消失 。 





如 果 ServerSocket 创 建 失败 ， 则 再 一 次 通过 main0 掷 出 违例 。 如 果 成 功 ， 
则 位 于 外 层 的 try-finally 代 码 块 可 以 担保 正确 的 清除 。 位 于 内 层 的 try- 
catch 块 只 负责 防范 ServeOneJabber 构 建 右 的 失败 ， 寿 构建 嚣 成功， 则 
ServeOneJabber 线 程 会 将 对 应 的 套 接 字 关 挥 。 


为 了 证 实 服务 器 代码 确实 能 为 多 名 客户 提供 服务 ， 下 面 这 个 程序 将 创建 
许多 客户 《使 用 线程 ) ， 并 同 相 同 的 服务 器 建立 连接 。 每 个 线程 的 “ 存 
在 时 间 ” 都 是 有 限 的 。 一 旦 到 期 ， 束 留 出 空间 以 便 创建 一 个 新 线程 。 允 
许 创建 的 线程 的 最 大 数量 是 由 final int maxthreads 决 定 的 。 大 家 会 注意 到 
这 个 值 非常 关键 ， 因 为 假如 把 它 设 得 很 大 ， 线 程 便 有 可 能 耗 尽 资 源 ， 并 
产生 不 可 预知 的 程序 错误 。 


//: MultiJabberClient.java 

















// Client that tests the MultiJabberServer 
// by starting up multiple clients. 
import java.net.*; 
import java.io.*; 
class JabberClientThread extends Thread { 
private Socket socket; 
private BufferedReader in; 
private PrintwWriter out; 
private static int counter = 0; 
private int id = counter++; 
private static int threadcount = 0; 
public static int threadCount() { 


return threadcount; 


public JabberClientThread(InetAddress addr) { 

System.out.println("Making client " + id); 
threadcount++; 
try { 

socket = 

new Socket(addr, MultiJabberServer.PORT); 

} catch(IOException e) { 

// If the creation of the socket fails, 


// nothing needs to be cleaned up. 


上 
try { 
in = 
new BufferedReader ( 
new InputStreamReader ( 
socket.getInputStream())); 
// Enable auto-flush: 
out = 
new PrintWriter ( 
new Bufferedwriter ( 
new OutputStreamwr iter ( 
socket.getOutputStream())), true); 
start(); 


} catch(IOException e) { 


// The socket should be closed on any 
// failures other than the socket 
// constructor: 
try { 
socket.close(); 
} catch(IOException e2) {} 
} 
// Otherwise the socket will be closed by 
// the run() method of the thread. 
} 
public void run() { 
try { 
for(int i = 0; i < 25; itt) { 
out.printin("Client " + id + ": " + i); 
String str = in.readLine(); 
System.out.println(str); 
} 
out.printin("END"); 
} catch(IOException e) { 
} finally { 
// Always close it: 


try { 


socket.close(); 


} catch(IOException e) {} 


threadcount--; // Ending this thread 


} 
public class MultiJabberClient { 
static final int MAX_THREADS = 40; 
public static void main(String[] args) 
throws IOException, InterruptedException { 
InetAddress addr = 
InetAddress.getByName(null); 
while(true) { 
if (JabberClientThread.threadCount( ) 
< MAX_THREADS ) 
new JabberClientThread(addr ); 


Thread.currentThread().sleep(100) ; 


} 
} ///:~ 


JabberClientThread 构 建 器 获取 一 个 InetAddress， 并 用 它 打开 一 个 套 接 
字 。 大 家 可 能 已 看 出 了 这 样 的 一 个 套路 : Socket 肯 定 用 于 创建 某 种 
Reader 以 及 / 或 者 Writer (或 者 InputStream 和 和 / OutputStream) 对 象 ， 
这 是 运用 Socket 的 唯一 方式 〈 当 然 ， 我 们 可 考虑 编写 一 、 两 个 类 ， 令 其 
自动 完成 这 些 操作 ， 避 免 大 量 重 复 的 代码 编写 工作 ) 。 同 样 地 ，start() 





执行 线程 的 初始 化 ， 并 调用 run0。 在 这 里 ， 消 息 发 送 给 服务 器 ， 而 来 自 
服务 器 的 信息 则 在 屏幕 上 回 显 出 来 。 然 而 ， 线 程 的 “存在 时 间 ” 是 有 限 

的 ， 最 终 都 会 结束 。 注 意 在 套 接 字 创 建 好 以 后 ， 但 在 构建 器 完成 之 前 ， 
假 知 构建 器 失败 ， 套 接 字 会 被 清除 。 人 否则 ， 为 套 接 字 调用 close0 的 责任 
便 落 到 了 run(0) 方 法 的 头 上 。 


threadcount 跟 踪 计 算 目 前 存在 的 JabberClientThread 对 象 的 数量 。 它 将 作 
为 构建 器 的 一 部 分 增值 ， 并 在 run0) 退 出 时 减 值 (run0 退 出 意味 着 线程 中 
止 》。 在 MultiJabberClient.main() 中 ， 大 家 可 以 看 到 线程 的 数量 会 得 到 检 
查 。 寿 数量 太 多 ， 则 多 余 的 暂时 不 创建 。 方 法 随后 进入 “休眠 ”状态 。 这 
样 一 来 ， 一 旦 部 分 线程 最 后 被 中 止 ， 多 作 的 那些 线程 就 可 以 创建 了 。 大 
家 可 试验 一 下 逐渐 增 大 MAX_THREADS， 看 看 对 于 你 使 用 的 系统 来 

说 ， 建 立 多 少 线程 〈 连 接 ) 才 会 使 您 的 系统 资源 降低 到 危险 程度 。 

















15.4 数据 报 


大 家 迄今 看 到 的 例子 使 用 的 都 是 “传输 控制 协议 ”(〈TCP) ， 亦 称 作 “基于 
数据 流 的 套 接 字 ”。 根 据 该 协议 的 设计 守则， 它 具 有 口上 度 的 可 靠 性 ， 而 
且 能 保证 数据 顺利 抵达 目的 地 。 换 言 之 ， 它 允许 重 传 那些 由 于 各 种 原因 
半路 “走失 ”的 数据 。 而 且 收 到 字 节 的 顺序 与 它们 发 出 来 时 是 一 样 的 。 当 
of 这 种 控制 与 可 靠 性 再 要 我 们 付出 一 些 代价 : TCP 具 有 非常 高 的 开 


还 有 另 一 种 协议 ， 名 为 “用 户 数据 报 协 议 ”(UDP) ， 它 并 不 刻意 追求 数 
据 包 会 完全 发 送出 去 ， 也 不 能 担保 它们 抵达 的 顺序 与 它们 发 出 时 一 样 。 
我 们 认为 这 是 一 种 “不 可 靠 协议 ”(TCP 当 然 是 “可 靠 协议 ”) 。 听 起 来 似 
乎 很 糟 ， 但 由 于 它 的 速度 快 得 多 ， 所 以 经 常 还 是 有 用 武之 地 的 。 对 某 些 
应 用 来 说 ， 比 如 声音 信号 的 传输 ， 如 果 少 量 数据 包 在 半路 上 丢失 了 ， 那 
么 用 不 着 太 在 意 ， 因 为 传输 的 速度 显得 更 重要 一 些 。 大 多 数 互 联网 游 
戏 ， 如 Diablo， 采 用 的 也 是 UDP 协议 通信 ， 因 为 网 络 通 信 的 快慢 是 游戏 
是 否 流畅 的 决定 性 因素 。 也 可 以 想 想 一 台 报时 服务 器 ， 如 果 某 条 消息 技 
失 了 ， 那 么 也 真 的 不 必 过 份 紧张 。 另 外 ， 有 些 应 用 也 许 能 向 服务 器 传 回 
An 
ie 会 o 


Java 对 数据 报 的 支持 与 它 对 TCP 套 接 字 的 文 持 大 致 相同 ， 但 也 存在 一 个 
明显 的 区 别 。 对 数据 报 来 说 ， 我 们 在 客户 和 服务 器 程序 都 可 以 放置 一 个 
DatagramSocket (数据 报 套 接 字 ) ， 但 与 ServerSocket 不 同 ， 前 者 不 会 干 
巴巴 地 等 竺 建立 一 个 连接 的 请 求 。 这 是 由 于 不 再 存在 “连接 ”， 取 而 代 之 
的 是 一 个 数据 报 陈 列 出 来 。 另 一 项 本 质 的 区 别 的 是 对 TCP 套 接 字 来 说 ， 

一 旦 我 们 建 好 了 连接 ， 便 不 再 需要 关心 谁 同 谁 “ 说 话 ” 一 一 只 需 通 过 会 话 
流 来 回 传送 数据 即 可 。 但 对 数据 报 来 说， 它 的 数据 包 必 须知 道 自己 来 自 
何 处 ， 以 及 打算 去 哪里 。 这 童 味 着 我 们 必须 知道 每 个 数据 报 包 的 这 些 信 
轧 ， 人 否则 信息 就 不 能 正常 地 传递 。 


DatagramSocket 用 于 收发 数据 包 ， 而 DatagramPacket 包 含 了 具体 的 信 

轧 。 准 备 接 收 一 个 数据 报时 ， 只 需 提 供 一 个 缓冲 区 ， 以 便 安 置 接收 到 的 
数据 。 数 据 包 抵达 时 ， 通 过 DatagramSocket， 作 为 信息 起 源 地 的 因特网 
地 址 以 及 端口 编写 会 自动 得 到 初 化 。 所 以 一 个 用 于 接收 数据 报 的 
DatagramPacket 构 建 器 是 : 









































DatagramPacket(buf, buf.length) 


其 中 ，buf 是 一 个 字 节 数组 。 既 然 buf 是 个 数组 ， 大 家 可 能 会 奇怪 为 什么 
构建 铬 目 己 不 能 调查 出 数组 的 长 度 呢 ? 实际 上 我 也 有 同感 ， 唯 一 能 猜 到 
人 





可 以 重复 使 用 数据 报 的 接收 代码 ， 不 必 每 次 都 建 一 个 新 的 。 每 次 用 它 的 
Hik Æ) ， 绥 冲 区 内 的 数据 都 会 被 履 普 。 


绥 冲 区 的 最 大 容量 仅 受 限于 允许 的 数据 报 包 大 小 ， 这 个 限制 位 于 比 
64KB 稍 小 的 地 方 。 但 在 许多 应 用 程序 中 ， 我 们 都 宁愿 它 变 得 还 要 小 一 
7a a 
特定 要 求 。 


发 出 一 个 数据 报时 ，DatagramPacket 不 仅 需 要 包含 正式 的 数据 ， 也 要 包 
含 因特网 地 址 以 及 端口 号 ， 以 决定 它 的 目的 地 。 上 所 以 用 于 输出 
DatagramPacket 的 构建 器 是 : 














DatagramPacket(buf, length, inetAddress, port) 


这 一 次 ，buf〈 一 个 字 节 数组 ) 已 经 包含 了 我 们 想 发 出 的 数据 。lengh 可 
以 是 buf 的 长 度 ， 但 也 可 以 更 短 一 些 ， 意 味 着 我 们 只 想 发 出 那么 多 的 字 
节 。 另 两 个 参数 分 别 代 表 数 据 包 要 到 达 的 因特网 地 址 以 及 目标 机 器 的 一 
个 目标 端口 〈 注 释 @) 。 


©: 我 们 认为 TCP 和 UDP 端口 是 相互 独立 的 。 也 就 是 说 ， 可 以 在 端口 
8080 同 时 运行 一 个 TCP 和 UDP 服 务 程序 ， 两 者 之 间 不 会 产生 冲突 。 


大 家 也 许 认 为 两 个 构建 器 创建 了 两 个 不 同 的 对 象 : 一 个 用 于 接收 数据 

报 ， 另 一 个 用 于 发 送 它们 。 如 果 是 好 的 面 癌 对 象 的 设计 方案 ， 会 建议 把 
它们 创建 成 两 个 不 同 的 类 ， 而 不 是 具有 不 同 的 行为 的 一 个 类 (具体 行为 
取决 于 我 们 如 何 构 建 对 象 ) 。 这 也 许 会 成 为 一 个 严重 的 问题 ， 但 幸运 的 
是 ，DatagramPacket 的 使 用 相当 人 简单， 我 们 不 需要 在 这 个 问题 上 纠缠 不 
清 。 这 一 点 在 下 例 里 将 有 很 明确 的 说 明 。 该 例 类 似 于 前 面 针 对 TCP 套 接 
字 的 MultiJabberServer 和 MultiJabberClient 例 子 。 多 个 客户 都 会 将 数据 报 
发 给 服务 器 ， 后 者 会 将 其 反馈 回 最 初 发 出 消息 的 同样 的 客户 。 














为 简化 从 一 个 String 里 创建 DatagramPacket 的 工作 〈 或 者 从 
DatagramPacket 里 创建 String) ， 这 个 例子 首先 用 到 了 一 个 工具 类 ， 名 为 
Dgram: 


//: Dgram.java 


// A utility class to convert back and forth 
// Between Strings and DataGramPackets. 
import java.net.*; 
public class Dgram { 
public static DatagramPacket toDatagram( 
String s, InetAddress destIA, int destPort) { 
// Deprecated in Java 1.1, but it works: 
byte[] buf = new byte[s.length() + 1]; 
s.getBytes(0, s.length(), buf, 0); 
// The correct Java 1.1 approach, but it's 
// Broken (it truncates the String): 
// byte[] buf = s.getBytes(); 
return new DatagramPacket(buf, buf.length, 
destIA, destPort); 
} 
public static String toString(DatagramPacket p){ 
// The Java 1.0 approach: 
// return new String(p.getData(), 


// ©, ©, p.getLength()); 


// The Java 1.1 approach: 
return 
new String(p.getData(), 0, p.getLength()); 


} 
} ///:~ 


Dgram 的 第 一 个 方法 采用 一 个 String、 一 个 InetAddress 以 及 一 个 端口 写作 
为 自己 的 参数 ， 将 String 的 内 容 复制 到 一 个 字 节 缓冲 区 ， 再 将 缓冲 区 传 
递 进 入 DatagramPacket 构 建 问 ， 从 而 构建 一 个 DatagramPacket。 注 意 绥 
冲 区 分 配 时 的 "+1" 一 一 这 对 防止 截 尾 现 象 是 非常 重要 的 。String 的 
getByte() 方 法 属于 一 种 特殊 操作 ， 能 将 一 个 字 串 包含 的 char 复 制 进 入 一 
个 字 市 缓冲 。 该 方法 现在 已 被 “反对 ”使 用 ;Java 1.1 有 一 个 “更 好 ”的 办 法 
来 做 这 个 工作 ， 但 在 这 里 却 被 当 作 注释 屏 菩 掉 了 ， 因 为 它 会 截 挥 String 
的 部 分 内 容 。 所 以 尽管 我 们 在 Java 1.1 下 编译 该 程序 时 会 得 到 一 条 “有 反 
对 ”消息 ， 但 它 的 行为 仍然 是 正确 无 误 的 〈 这 个 错误 应 该 在 你 读 到 这 里 
的 时 候 修正 了 ) 。 


Dgram.toString() 方 法 同时 展示 了 Java 1.0 的 方法 和 Java 1.1 的 方法 (两 者 
是 不 同 的 ， 因 为 有 一 种 新 类 型 的 String 构 建 器 ) 。 


下 面 是 用 于 数据 报 演示 的 服务 占 代 码 : 


//: ChatterServer.java 











// A server that echoes datagrams 
import java.net.*; 

import java.io.*; 

import java.util.*; 

public class ChatterServer { 


static final int INPORT = 1711; 


private byte[] buf = new byte[1000]; 
private DatagramPacket dp = 
new DatagramPacket(buf, buf.length); 
// Can listen & send on the same socket: 
private DatagramSocket socket; 
public ChatterServer() { 
try { 
socket = new DatagramSocket(INPORT); 
System.out.println("Server started"); 
while(true) { 
// Block until a datagram appears: 
socket.receive(dp); 
String rcvd = Dgram.toString(dp) + 
", from address: " + dp.getAddress() + 
", port: " + dp.getPort(); 
System.out.println(rcvd); 
String echoString = 
"Echoed: " + rcvd; 
// Extract the address and port from the 
// received datagram to find out where to 
// send it back: 
DatagramPacket echo = 


Dgram.toDatagram(echoString, 


dp.getAddress(), dp.getPort()); 
socket.send(echo); 


} 

} catch(SocketException e) { 
System.err.println("Can't open socket"); 
System.exit(1); 

} catch(IOException e) { 

System.err.printin( "Communication error"); 


e.printStackTrace(); 


} 
public static void main(String[] args) { 


new ChatterServer(); 


} 
NS fief E 


ChatterServer 创 建 了 一 个 用 来 接收 消息 的 DatagramSocket〈 数 据 报 套 接 
字 ) ， 而 不 是 在 我 们 每 次 准备 接收 一 条 新 消息 时 都 新 建 一 个 。 这 个 单一 
的 DatagramSocket 可 以 重复 使 用 。 它 有 一 个 端口 号 ， 因 为 这 属于 服务 
器 ， 客 户 必须 确切 知道 自己 把 数据 报 发 到 哪个 地 址 。 尽 管 有 一 个 端口 
号 ， 但 没有 为 它 分 配 因特网 地 址 ， 因 为 它 束 驻 留 在 “这 ” 台 机 器 内 ， 所 以 
知道 目 己 的 因特网 地 址 是 什么 《目前 是 默认 的 localhost) 。 在 无 限 while 
循环 中 ， 套 接 字 被 告知 接收 数据 (receive0) 。 然 后 暂时 挂 起 ， 直 到 一 
个 数据 报 出 现 ， 再 把 它 反馈 回 我 们 希望 的 接收 人 一 一 DatagramPacket dp 
一 一 里 面 。 数 据 包 〈Packet) 会 被 转换 成 一 个 字 串 ， 同 时 插入 的 还 有 数 
据 包 的 起 源 因 特 网 地 址 及 套 接 字 。 这 些 信息 会 显示 出 来 ， 然 后 添加 一 个 
额外 的 字 串 ， 指 出 自己 已 从 服务 器 反馈 回来 了 。 














大 家 可 能 会 觉得 有 点 儿 迷 惑 。 正 如 大 家 会 看 到 的 那样 ， 许 多 不 同 的 因 特 
网 地 址 和 端口 号 都 可 能 是 消息 的 起 源 地 一 一 换言之 ， 客 户 程序 可 能 驻 留 
在 任何 一 台 机 器 里 (就 这 一 次 演示 来 说 ， 它 们 都 驻 留 在 localhost 里 ， 但 
每 个 客户 使 用 的 端口 编写 是 不 同 的 ) 。 为 了 将 一 条 消息 送 回 它 真 正 的 始 
发 客户 ， 需 要 知道 那个 客户 的 因特网 地 址 以 及 端口 号 。 笠 运 的 是 ， 所 有 
这 些 资 料 均 已 非常 周到 地 封装 到 发 出 消息 的 DatagramPacket 内 部 ， 所 以 
我 们 要 做 的 全 部 事情 就 是 用 getAddress0 和 getPortO 把 它们 取出 来 。 利 用 
这 些 资料 ， 可 以 构建 DatagramPacket echo 它 通过 与 接收 用 的 相同 的 
套 接 字 发 送 回来 。 除 此 以 外 ， 一 旦 套 接 字 发 出 数据 报 ， 就 会 添加 “这 ”人 台 
机 器 的 因特网 地 址 及 端口 信息 ， 所 以 当 客 户 接收 消息 时 ， 它 可 以 利用 
getAddress() 和 getPort() 了 解数 据 报 来 自 何 处 。 事 实 上 ，getAddress() 和 
getPort() 唯 一 不 能 告诉 我 们 数据 报 来 自 何 处 的 前 提 是 : 我 们 创建 一 个 竺 
发 送 的 数据 报 ， 并 在 正式 发 出 之 前 调用 了 getAddress0 和 getPortD。 到 数 
据 报 正式 发 送 的 时 候 ， 这 人 台 机 器 的 地 址 以 及 端口 才 会 号 入 数据 报 。 所 以 
我 们 得 到 了 运用 数据 报时 一 项 重要 的 原则 : 不 必 跟 踪 一 条 消息 的 来 源 
地 ! 因为 它 表 定 保存 在 数据 报 里 。 事 实 上 ， 对 程序 来 说 ， 最 可 靠 的 做 法 
是 我 们 不 要 试图 跟踪， 而 是 无 论 如 何 都 从 目标 数据 报 里 提取 出 地 址 以 及 
端口 信息 《就 象 这 里 做 的 那样 ) 。 


为 测试 服务 器 的 运转 是 人 否 正 常 ， 下 面 这 程序 将 创建 大 量 客户 《〈 线 程 ) ， 
它们 都 会 将 数据 报 包 发 给 服务 器 ， 并 等 候 服 务 器 把 它们 原样 反馈 回来 。 


//: ChatterServer.java 



































// A server that echoes datagrams 
import java.net.*; 
import java.io.*; 
import java.util.*; 
public class ChatterServer { 
static final int INPORT = 1711; 
private byte[] buf = new byte[1000]; 


private DatagramPacket dp = 


new DatagramPacket(buf, buf.length); 
// Can listen & send on the same socket: 
private DatagramSocket socket; 
public ChatterServer() { 
try { 
socket = new DatagramSocket(INPORT); 
System.out.printin("Server started"); 
while(true) { 
// Block until a datagram appears: 
socket.receive(dp); 
String rcvd = Dgram.toString(dp) + 
"| from address: " + dp.getAddress() + 
", port: " + dp.getPort(); 
System.out.println(rcvd); 
String echoString = 
"Echoed: " + rcvd; 
// Extract the address and port from the 
// received datagram to find out where to 
// send it back: 
DatagramPacket echo = 
Dgram.toDatagram(echoString, 
dp.getAddress(), dp.getPort()); 


socket.send(echo); 


} 

} catch(SocketException e) { 
System.err.println("Can't open socket"); 
System.exit(1); 

} catch(IOException e) { 

System.err.printin( "Communication error"); 


e.printStackTrace(); 


} 
public static void main(String[] args) { 


new ChatterServer(); 


} 
TTT 


ChatterClient 被 创建 成 一 个 线程 (Thread) ， 所 以 可 以 用 多 个 客户 来 “ 驿 
扰 ?服务 器 。 从 中 可 以 看 到 ， 用 于 接收 的 DatagramPacket 和 用 于 
ChatterServer 的 那个 是 相似 的 。 在 构建 器 中 ， 创 建 DatagramPacket 时 没有 
附带 任何 参数 〈 自 变量 ) ， 因 为 它 不 需要 明确 指出 自己 位 于 哪个 特定 编 
号 的 端口 里 。 用 于 这 个 套 接 字 的 因特网 地 址 将 成 为 "这 人 台 机 器 >”《〈 比 如 
localhost) ， 而 且 会 自动 分 配 问 口 编号 ， 这 从 输出 结果 即 可 看 出 。 同 用 
于 服务 器 的 那个 一 样 ， 这 个 DatagramPacket 将 同时 用 于 发 送 和 接收 。 


hostAddress 是 我 们 想 与 之 通信 的 那 台 机 器 的 因特网 地 址 。 在 程序 中 ， 如 
果 需 要 创建 一 个 准备 传 出 去 的 DatagramPacket， 那 么 必须 知道 一 个 准确 
的 因特网 地 址 和 端口 号 。 可 以 表 定 的 是 ， 主 机 必须 位 于 一 个 已 知 的 地 址 
和 端口 号 上 ， 使 客户 能 启动 与 主机 的 “会 话 ”。 


每 个 线程 都 有 自己 独一无二 的 标识 号 《尽管 自动 分 配给 线程 的 端口 号 是 
也 会 提供 一 个 唯一 的 标识 符 ) 。 在 run0 中 ， 我 们 创建 了 一 个 String 消 





旦 ， 其 中 包含 了 线程 的 标识 编写 以 及 该 线程 准备 发 送 的 消 因 编写。 我 们 
用 这 个 字 串 创建 一 个 数据 报 ， 发 到 主机 上 的 指定 地 址 ; 端口 编号 则 直接 
从 ChatterServer 内 的 一 个 第 数 取 得 。 一 旦 消息 发 出 ，receive0 就 会 暂时 
被 “堵塞 起来， 直到 服务 器 回复 了 这 条 消息 。 与 消息 附 在 一 起 的 所 有 信 
上 县 使 我 们 知道 回 到 这 个 特定 线程 的 东西 正 是 从 始 发 消息 中 投递 出 去 的 。 
在 这 个 例子 中 ， 尽 管 是 一 种 “不 可 靠 ? 协 议 ， 但 仍然 能 够 检查 数据 报 是 否 
到 去 过 了 它们 该 去 的 地 方 〈 这 在 localhost 和 LAN 环 境 中 是 成 立 的 ， 但 在 
非 本 地 连接 中 却 可 能 出 现 一 些 错误 ) o 


运行 该 程序 时 ， 大 家 会 发 现 每 个 线程 都 会 结束 。 这 意味 着 发 送 到 服务 器 


的 每 个 数据 报 包 都 会 回转 ， 并 反馈 回 正 确 的 接收 者 。 如 果 不 是 这 样 ， 一 
E 














大 家 或 许 认为 将 文件 从 一 台 机 器 传 到 另 一 台 的 唯一 正确 方式 是 通过 TCP 
套 接 字 ， 因 为 它们 是 “可 靠 的 。 然 而 ， 由 于 数据 报 的 速度 非常 快 ， 所 以 
它 才 是 一 种 更 好 的 选择 。 我 们 只 需 将 文件 分 割 成 多 个 数据 报 ， 并 为 每 个 
包 编号 。 接 收 机 器 会 取得 这 些 数据 包 ， 并 重新 “组 装 ” 它 们 ， 一 个 “标题 
包 ” 会 告诉 机 器 应 该 接收 多 少 个 包 ， 以 及 组 装 所 需 的 另 一 些 重要 信息 。 
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15.5 一 个 Web 应 用 


现在 让 我 们 想 想 如 何 创建 一 个 应 用 ， 令 其 在 真实 的 Web 环 境 中 运行 ， 它 
将 把 Java 的 优势 表现 得 淋漓 尺 臻 。 这 个 应 用 的 一 部 分 是 在 Web 服 务 器 上 
运行 的 一 个 Java 程 序 ， 男 一 部 分 则 是 一 个 “程序 片 ” 或 “小 应 用 程 

序 ”(Applet) ， 从 服务 器 下 载 至 浏览 器 《〈 即 “客户 ”) 。 这 个 程序 片 从 用 
户 那 里 收集 信息 ， 并 将 其 传 回 Web 服 务 右 上 运行 的 应 用 程序 。 程 序 的 任 
务 非常 简单 : 程序 片 会 询问 用 户 的 E-mail] 地 址 ， 并 在 验证 这 个 地 址 合格 
后 (没有 包含 空格 ， 而 且 有 一 个 @ 符 写 ) ， 将 该 E-mail 发 送 给 Web 服 务 
器 。 服 务 器 上 运行 的 程序 则 会 捕获 传 回 的 数据 ， 检 查 一 个 包含 了 所 有 E- 
mail 地 址 的 数据 文件 。 如 果 那 个 地 址 已 包含 在 文件 里 ， 则 向 浏览 器 反馈 
一 条 消息 ， 说 明 这 一 情况 。 该 消息 由 程序 片 负责 显示 。 知 是 一 个 新 地 
址 ， 则 将 其 置 入 列表 ， 并 通知 程序 片 已 成 功 添 加 了 电子 函件 地 址 。 


石 采 用 传统 方式 来 解决 这 个 问题 ， 我 们 要 创建 一 个 包含 了 文本 字段 及 一 
AFE” (Submit) 按钮 的 HTML 页 。 用 户 可 在 文本 字段 里 键入 目 己 喜 
欢 的 任何 内 容 ， 并 坚 无 阻碍 地 提交 给 服务 器 《在 客户 端 不 进行 任何 检 

A) 。 提 区 数据 的 同时 ，Web 页 也 会 告诉 服务 器 应 对 数据 采取 什么 样 的 
操作 一 一 知 会 “通用 网 关 接 口 ”(CGI) 程序 ， 收 到 这 些 数据 后 立即 运行 
服务 器 。 这 种 CGI 程序 通常 是 用 Pen 或 C 写 的 《有 时 也 用 C++， 但 要 求 服 
Fae CH) ， 而 且 必 须 能 控制 一 切 可 能 出 现 的 情况 。 它 首先 会 检查 数 

据 ， 判 断 是 否 及 用 了 正确 的 格式 。 大 答案 是 否定 的 ， 则 CGI 程 序 必须 创 
建 一 个 HTML 页， 对 遇 到 的 问题 进行 描述 。 这 个 页 会 转交 给 服务 器 ， 再 
由 服务 器 反馈 回 用 户 。 用 户 看 到 出 错 提示 后 ， 必 须 再 试 一 过 提交 ， 直 到 
通过 为 止 。 知 数据 正确 ，CGI 程 序 会 打开 数据 文件 ， 要 么 把 电子 函件 地 
址 加 入 文件 ， 要 么 指出 该 地 址 已 在 数据 文件 里 了 。 无 论 哪 种 情况 ， 都 必 
须 格式 化 一 个 恰当 的 HTML 页 ， 以 便服 务 器 返回 给 用 户 。 


作为 Java 程 序 员 ， 上 述 解决 问题 的 方法 显得 非常 沦 措 。 而 且 很 目 然 地 ， 
我 们 希望 一 切 工作 都 用 Java 完 成 。 首 先 ， 我 们 会 用 一 个 Java 程 序 片 负责 
客户 端的 数据 有 效 性 校 验 ， 避 免 数 据 在 服务 占 和 客户 之 间 传 来 传 去 ， 浪 
费时 间 和 带 党 ， 同 时 减轻 服务 占 额 外 构建 HTML 页 的 人 负担。 然后 跳 过 
Perl CGI 脚本 ， 换 成 在 服务 器 上 运行 一 个 Java 应 用 。 事 实 上 ， 我 们 在 这 
儿 已 完全 跳 过 了 了 Web 服务器 ， 仅 仅 需 要 从 程序 片 到 服务 器 上 运行 的 Java 
应 用 之 间 建 立 一 个 连接 即 可 。 


























正如 大 家 不 久 束 会 体验 到 的 那样 ， 尽 管 看 起 来 非常 简单 ， 但 实际 上 有 一 
些 意 想不到 的 问题 使 局 面 显 得 稍微 有 些 复杂 。 用 Java 1.1 写 程序 片 是 最 
理想 的 ， 但 实际 上 却 经 常 行 不 通 。 到 本 书写 作 的 时 候 ， 拥 有 Java 1.1 能 
力 的 浏览 器 仍 为 数 不 多 ， 而 且 即 使 这 类 浏览 器 现在 非常 流行 ， 仍 需 考 上 处 
照顾 一 下 那些 升级 缓慢 的 人 。 所 以 从 安全 的 角度 看 ， 程 序 片 代码 最 好 只 
用 Java 1.0 编 写 。 基 于 这 一 前 提 ， 我 们 不 能 用 JAR 文 件 来 合并 (压缩 ) FE 
序 片 中 的 .class 文 件 。 所 以 ， 我 们 应 尽 可 能 减少 .class 文 件 的 使 用 数量 ， 
以 缩短 下 载 时 间 。 


好 了 ， 再 来 说 说 我 用 的 Web 服 务 器 《〈 写 这 个 示范 程序 时 用 的 就 是 它 ) 。 
它 确 实 文 持 Java， 但 仅 限 于 Java 1.0! 所 以 服务 器 应 用 也 必须 用 Java 1.0 
编写 。 


15.5.1 服务 器 应 用 


现在 讨论 一 下 服务 喜 应 用 《程序 ) 的 问题 ， 我 把 它 叫 作 

NameCollecor (4 FWRI) 。 假 如 多 名 用 户 同 时 尝试 提交 他 们 的 E- 

mail 地 址 ， 那 么 会 发 生 什 么 情况 呢 ? 知 NameCollector 使 用 TCP/IP 套 接 

字 ， 那 么 必须 运用 早先 介绍 的 多 线程 机 制 来 实现 对 多 个 客户 的 并 发 控 

制 。 但 所 有 这 些 线程 都 试图 把 数据 写 到 同一 个 文件 里 ， 其 中 保存 了 所 有 
E-mail 地 址 。 这 便 要 求 我们 设立 一 种 锁定 机 制 ， 保 证 多 个 线程 不 会 同时 
访问 那个 文件 。 一 个 “信号 机 ”可 在 这 里 帮助 我 们 达到 目的 ， 但 或 许 还 有 
一 种 更 简单 的 方式 。 


如 宁 我 们 换 用 数据 报 ， 束 不 必 使 用 多 线程 了 。 用 单个 数据 报 即 可 “ 侦 
听 ” 进 入 的 所 有 数据 报 。 一 旦 监视 到 有 进入 的 消 恩 ， 程 序 束 会 进行 适当 
的 处 理 ， 并 将 答复 数据 作为 一 个 数据 报 传 回 原 先 发 出 请 求 的 那 名 接收 
者 。 右 数据 报 半路 上 丢失 了 ， 则 用 户 会 注意 到 没有 答复 数据 传 回 ， 所 以 
可 以 重新 提交 请 求 。 


服务 器 应 用 收 到 一 个 数据 报 ， 并 对 它 进行 解读 的 时 候 ， 必 须 提 取出 其 中 
的 电子 函件 地 址 ， 并 检查 本 机 保存 的 数据 文件 ， 看 看 里 面 是 否 已 经 包含 
了 那个 地 址 〈 如 果 没 有 ， 则 添加 之 ) 。 所 以 我 们 现在 遇 到 了 一 个 新 的 问 
jl. Java “1.0 似乎 没有 足够 的 能 力 来 方便 地 处 理 包含 了 电子 函件 地 址 的 
XF (Java 1.1 则 不 然 ) 。 但 是 ， 用 C 轻 易 束 可 以 解决 这 个 问题 。 因 此 ， 
我 们 在 这 儿 有 机 会 学 习 将 一 个 非 Java 程 序 同 Java 程 序 连 接 的 最 简便 方 

式 。 程 序 使 用 的 Runtime 对 象 包 含 了 一 个 名 为 exec() 的 方法 ， 它 会 独立 机 
器 上 一 个 独立 的 程序 ， 并 返回 一 个 Process〈 进 程 ) 对 象 。 我 们 可 以 取得 




















一 个 OutputStream， 它 同 这 个 单独 程序 的 标准 输入 连接 在 一 起 ; 并 取得 
一 个 InputStream， 它 则 同 标 准 输出 连接 到 一 起 。 要 做 的 全 部 事情 就 是 用 
任何 语言 写 一 个 程序 ， 只 要 它 能 从 标准 输入 中 取得 自己 的 输入 数据 ， 并 
将 输出 结果 写 入 标准 输出 即 可 。 如 果 有 些 问 题 不 能 用 Java 人 简便 与 快速 地 
解决 (或 者 想 利用 原 有 代码 ， 不 想 改写 ) ， 就 可 以 考虑 采用 这 种 方法 。 
亦 可 使 用 Java 的 “固有 方法 ”(Native Method) ， 但 那 要 求 更 多 的 技巧 ， 
大 家 可 以 参考 一 下 附录 A。 


1. C 程 序 


这 个 非 Java 应 用 是 用 C 写 成 ， 因 为 Java 不 适合 作 CGI 编 程 ， 起 码 启 动 的 时 
间 不 能 让 人 满意 。 它 的 任务 是 管理 电子 函件 (E-mail 地址 的 一 个 列 
表 。 标 准 输入 会 接受 一 个 E-mail 地 址 ， 程 序 会 检查 列表 中 的 名 字 ， 判 断 
是 否 存在 那个 地 址 。 知 不 存在 ， 就 将 其 加 入 ， 并 报告 操作 成 功 。 但 假如 
名 字 已 在 列表 里 了 ， 就 需要 指出 这 一 点 ， 避 人 免 重复 加 入 。 大 家 不 必 担 心 
自己 不 能 完全 理解 下 列 代 码 的 含义 。 它 仅仅 是 一 个 演示 程序 ， 告 诉 你 如 
何 用 其 他 语言 写 一 个 程序 ， 并 从 Java 中 调用 它 。 在 这 里 具体 采用 何 种 语 
aa 只 要 能 够 从 标准 输入 中 读 取 数据 ， 并 能 写 入 标准 输出 即 
Hjo 


//: Listmgr.c 











// Used by NameCollector.java to manage 

// the email list file on the server 

#include <stdio.h> 

#include <stdlib.h> 

#include <string.h> 

#define BSIZE 250 

int alreadyInList(FILE* list, char* name) { 
char lbuf[BSIZE]; 


// Go to the beginning of the list: 


fseek(list, ©, SEEK_SET); 
// Read each line in the list: 
while(fgets(lbuf, BSIZE, list)) { 
// Strip off the newline: 
char * newline = strchr(lbuf, '\n'); 
if(newline != 0) 
*newline = '\O'; 
if(strcemp(lbuf, name) == 0) 
return 1; 


} 


return 0; 
Í 
int main() { 
char buf[BSIZE]; 
FILE* list = fopen("emlist.txt", "a+t"); 
if(list == 0) { 
perror("could not open emlist.txt"); 
exit(1); 
} 
while(1) { 
gets(buf); /* From stdin */ 
if(alreadyInList(list, buf)) { 


printf("Already in list: %s", buf); 


fflush(stdout); 

t 

else { 
fseek(list, 0, SEEK_END); 
fprintf(list, "%s\n", buf); 
fflush(list); 
printf("%s added to list", buf); 


fflush(stdout); 


} 
} ///:~ 


该 程序 假设 C 编 译 器 能 接受 /样式 注释 (许多 编译 器 都 能 ， 亦 可 换 用 一 
个 C++ 编译 器 来 编译 这 个 程序 ) 。 如 果 你 的 编译 器 不 能 接受 ， 则 简单 地 
将 那些 注释 删 掉 即 可 。 


文件 中 的 第 一 个 函数 检查 我 们 作为 第 二 个 参数 〈 指 同一 个 char 的 指针 ) 
传递 给 它 的 名 字 是 否 已 在 文件 中 。 在 这 儿 ， 我 们 将 文件 作为 一 个 FILE 指 
针 传 递 ， 它 指向 一 个 已 打开 的 文件 (文件 是 在 main() 中 打开 的 ) . KA 
fseekO 在 文件 中 表 历 ; 我 们 在 这 儿 用 它 移 至 文件 开头 。fgets0 从 文件 list 
中 读 入 一 行内 容 ， 并 将 其 置 入 缓冲 区 lbuf 一 一 不 会 超过 规定 的 缓冲 区 长 
度 BSIZE。 上 所 有 这 些 工作 都 在 一 个 while 循 环 中 进行 ， 所 以 文件 中 的 每 一 
行 都 会 谈 入 。 接 下 来 ， 用 strchr0O 找 到 新 行 字 符 ， 以 便 将 其 删 掉 。 最 后 ， 
用 strcmpO 比 较 我 们 传递 给 函数 的 名 字 与 文件 中 的 当前 行 。 奉 找到 一 致 
的 内 容 ，strcmp0 会 返回 0。 画 数 随后 会 退出 ， 并 返回 一 个 1， 指 出 该 名 
字 已 经 在 文件 里 了 “注意 这 个 函数 找到 相符 内 容 后 会 立即 返回 ， 不 会 把 
时 间 浪 费 在 检查 列表 剩余 内 容 的 上 面 )。 如 果 找 遍 列 表 都 没有 发 现 相符 
的 内 容 ， 则 函数 返回 0。 


在 main0 中 ， 我 们 用 fopen0 打 开 文 件 。 第 一 个 参数 是 文件 名 ， 第 二 个 是 

















打开 文件 的 方式 ; a+ 表 示 “ 退 加 ”， 以 及 “打开 ”( 或 “创建 "， 假 有 文件 尚 
不 存在 )， 以 便 到 文件 的 末尾 进行 更 新 。fopen() 函 数 返回 的 是 一 个 FILE 
指针 ， 知 为 0， 表 示 打 开 操作 失败 。 此 时 需要 用 perror0 打 印 一 条 出 错 提 
示 消 息 ， 并 用 exit0 中 止 程序 运行 。 


如 果 文 件 成 功 打 开 ， 程 序 就 会 进入 一 个 无 限 循环 。 调 用 gets(buf) 的 函数 
会 从 标准 输入 中 取出 一 行 〈 记 住 标准 输入 会 与 Java 程 序 连 接 到 一 起 ) ， 
并 将 其 置 入 缓冲 区 buf 中 。 组 冲 区 的 内 容 随后 会 简单 地 传递 给 
alreadyInList() 水 数 ， 如 内 容 已 在 列表 中 ，printf0) 束 会 将 那 条 消 恩 及 给 标 
准 输出 Java 程序 正 在 监视 它 ) 。fflushO 用 于 对 输出 缓冲 区 进行 刷新 。 


如 果 名 字 不 在 列表 中 ， 束 用 fseek() 移 到 列表 末尾 ， 并 用 fprintf() 将 名 
字 “ 打 印 ” 到 列表 末尾 。 随 后 ， 用 printf() 指 出 名 字 已 成 功 加 入 列表 (同样 
需要 刷新 标准 和 输出) ， 无 限 循环 返回 ， 继 续 等 候 一 个 新 名 字 的 进入 。 


记 住 一 般 不 能 先 在 自己 的 计算 机 上 编译 此 程序 ， 再 把 编译 好 的 内 容 上 载 
到 Web 服 务 器 ， 因 为 那 台 机 器 使 用 的 可 能 是 不 同类 的 处 理 器 和 操作 系 
统 。 例 如 ， 我 的 web 服 务 器 安装 的 是 Intel 的 CPU， 但 操作 系统 是 Linux， 
所 以 必须 先 下 载 源 码 ， 再 用 远程 命令 〈 通 过 telnet) 指挥 Linux 上 自 带 的 C 
编译 器 ， 令 其 在 服务 器 端 编译 好 程序 。 


2. Java 程 序 
这 个 程序 先 启 动 上 述 的 C 程 序 ， 再 建立 必要 的 连接 ， 以 便 同 它 “ 交 谈 ”。 


随后 ， 它 创建 一 个 数据 报 套 接 字 ， 用 它 “ 监 视 ” 或 者 “ 侦 听 ”来 自 程序 片 的 
数据 报 包 。 


//: NameCollector.java 








// Extracts email names from datagrams and stores 
// them inside a file, using Java 1.02. 

import java.net.*; 

import java.io.*; 


import java.util.*; 


public class NameCollector { 
final static int COLLECTOR_PORT = 8080; 
final static int BUFFER_SIZE = 1000; 
byte[] buf = new byte[BUFFER_SIZE]; 
DatagramPacket dp = 
new DatagramPacket(buf, buf.length); 
// Can listen & send on the same socket: 
DatagramSocket socket; 
Process listmgr; 
PrintStream nameList; 
DataInputStream addResult; 
public NameCollector() { 
try { 
listmgr = 
Runtime.getRuntime().exec("listmgr.exe"); 
nameList = new PrintStream( 
new BufferedOutputStream( 
listmgr.getOutputStream())); 
addResult = new DataInputStream( 
new BufferedInputStream( 
listmgr.getInputStream())); 
} catch(IOException e) { 


System.err.printin( 


"Cannot start listmgr.exe"); 
System.exit(1); 
} 


try { 


socket = 
new DatagramSocket(COLLECTOR_PORT); 
System.out.printin( 
"NameCollector Server started"); 
while(true) { 
// Block until a datagram appears: 
socket.receive(dp); 
String rcvd = new String(dp.getData(), 
©, 0, dp.getLength()); 
// Send to listmgr.exe standard input: 
nameList.println(rcvd.trim()); 
nameList.flush(); 
byte[] resultBuf = new byte[BUFFER_SIZE]; 
int byteCount = 
addResult.read(resultBuf ) ; 
if(byteCount != -1) { 
String result = 
new String(resultBuf, 0).trim(); 


// Extract the address and port from 


// the received datagram to find out 
// where to send the reply: 
InetAddress senderAddress = 
dp.getAddress(); 
int senderPort = dp.getPort(); 
byte[] echoBuf = new byte[BUFFER_SIZE]; 
result.getBytes( 
©, byteCount, echoBuf, 0); 
DatagramPacket echo = 
new DatagramPacket ( 
echoBuf, echoBuf.length, 
senderAddress, senderPort); 
socket.send(echo); 
} 
else 
System.out.printin( 
"Unexpected lack of result from " + 
"listmgr.exe"); 
} 
} catch(SocketException e) { 
System.err.println("Can't open socket"); 
System.exit(1); 


} catch(IOException e) { 


System.err.println("Communication error"); 


e.printStackTrace(); 


} 
public static void main(String[] args) { 
new NameCollector(); 
} 
} ///:~ 





NameCollector 中 的 第 一 个 定义 应 该 是 大 家 所 熟悉 的 : 选 定 端口 ， 创 建 一 
个 数据 报 包 ， 然 后 创建 指 同 一 个 DatagramSocket 的 句柄 。 接 下 来 的 三 个 
定义 负责 与 C 程 序 的 连接 : 一 个 Process 对 象 是 C 程 序 由 Java 程 序 启动 之 后 
返回 的 ， 而 且 那 个 Process 对 象 产生 了 InputStreaam 和 OutputStream， 分 别 
代表 C 程 序 的 标准 输出 和 标准 输入 。 和 Java IO 一 样 ， 它 们 理所当然 地 需 
要 “封装 ?起 来 ， 所 以 我 们 最 后 得 到 的 是 一 个 PrintStream 和 


DataInputStream. 


这 个 程序 的 所 有 工作 都 是 在 构建 器 内 进行 的 。 为 启动 C 程 序 ， 需 要 取得 
当前 的 Runtime 对 象 。 我 们 用 它 调 用 exec()， 再 由 后 者 返回 Process 对 象 。 
在 Process 对 象 中 ， 大 家 可 看 到 通过 一 简单 的 调用 即 可 生成 数据 流 : 
getOutputStream() 和 getInputStream()。 从 这 个 时 候 开 始 ， 我 们 需要 考虑 
的 全 部 事情 就 是 将 数据 传 给 数据 流 nameList， 并 从 addResult 中 取得 结 
果 。 


和 往常 一 样 ， 我 们 将 DatagramSocket 同 一 个 端口 连接 到 一 起 。 在 无 限 
while 循 环 中 ， 程 序 会 调用 receive0) 除非 一 个 数据 报到 来 ， 人 否则 
receive() 会 一 起 处 于 “ 墙 塞 ” 状 态 。 数 据 报 出 现 以 后 ， 它 的 内 容 会 提取 到 
String rcvd 里 。 我 们 首先 将 该 字 串 两 头 的 空格 剔除 Crim) ， 再 将 其 发 给 
C 程 序 。 如 下 所 示 : 


























nameList.println(rcvd.trim()); 





之 所 以 能 这 样 编码 ， 是 因为 Java 的 execO 人 允许 我 们 访问 任何 可 执行 模 
块 ， 只 要 它 能 从 标准 输入 中 读 ， 并 能 同 标 准 输出 中 写 。 还 有 男 一 些 方式 
可 与 非 Java 代 码 “ 交 谈 ”"， 这 将 在 附录 A 中 讨论 。 


从 C 程 序 中 捕获 结果 束 显 得 稍微 及 烦 一 些 。 我 们 必须 调用 read0， 并 提供 
一 个 绥 冲 区 ， 以 便 保 存 结果 。read0 的 返回 值 是 来 自 C 程 序 的 字 节 数 。 知 
这 个 值 为 -1， 意 味 着 某 个 地 方 出 现 了 问题 。 否 则 ， 我 们 就 将 

resultBuf (REW) 转换 成 一 个 字 串 ， 然 后 同样 清除 多 余 的 空格 。 

随后 ， 这 个 字 串 会 象 往常 一 样 进入 一 个 DatagramPacket， 并 传 回 当初 发 
出 请 求 的 那个 同样 的 地 址 。 注 意 发 送 方 的 地 址 也 是 我 们 接收 到 的 


DatagramPacket 的 一 部 分 。 


记 住 尽 管 C 程 序 必 须 在 Web 服 务 器 上 编译 ， 但 Java 程 序 的 编译 场所 可 以 
是 任意 的 。 这 是 由 于 不 管 使 用 的 是 什么 硬件 平台 和 操作 系统 ， 编 译 得 到 
的 季节 人 码 都 是 一 样 的 。 束 束 是 Java 的 “ 跨 平 台 ” 兼 容 能 力 。 


15.5.2 NameSender 程 序 片 


正如 早先 指出 的 那样 ， 程 序 片 必须 用 Java 1.0 编写 ， 使 其 能 与 绝 大 多 数 
的 浏览 器 适应 。 也 正 是 由 于 这 个 原因 ， 我 们 产生 的 类 数量 应 尽 可 能 地 
少 。 所 以 我 们 在 这 儿 不 考虑 使 用 前 面 设 计 好 的 Dgram 类 ， 而 将 数据 报 的 
所 有 维护 工作 都 转 到 代码 行 中 进行 。 此 外 ， 程 序 片 要 用 一 个 线程 监视 由 
服务 器 传 回 的 响应 信息 ， 而 非 实 现 Runnable 接 口 ， 用 集成 到 程序 片 的 一 
个 独立 线程 来 做 这 件 事情 。 当 然 ， 这 样 做 对 代码 的 可 读 性 不 利 ， 但 却 能 
产生 一 个 单 类 《以 及 单个 服务 器 请 求 ) 程序 片 : 


//: NameSender.java 











// An applet that sends an email address 
// as a datagram, using Java 1.02. 
import java.awt.*; 

import java.applet.*; 

import java.net.*; 


import java.io.*; 


public class NameSender extends Applet 
implements Runnable { 
private Thread pl = null; 
private Button send = new Button( 
"Add email address to mailing list"); 
private TextField t = new TextField( 
"type your email address here", 40); 
private String str = new String(); 
private Label 
1 = new Label(), 12 = new Label(); 
private DatagramSocket s; 
private InetAddress hostAddress; 
private byte[] buf = 
new byte[NameCollector.BUFFER_SIZE]; 
private DatagramPacket dp = 
new DatagramPacket(buf, buf.length); 
private int vcount = 0; 
public void init() { 
setLayout(new BorderLayout()); 
Panel p = new Panel(); 
p.setLayout(new GridLayout(2, 1)); 
p.add(t); 


p.add(send); 


add("North", p); 
Panel labels = new Panel(); 
labels.setLayout(new GridLayout(2, 1)); 
labels.add(1l); 
labels.add(12); 
add("Center", labels); 
try { 
// Auto-assign port number: 
s = new DatagramSocket(); 
hostAddress = InetAddress.getByName( 
getCodeBase().getHost()); 
} catch(UnknownHostException e) { 
1l.setText("Cannot find host"); 
} catch(SocketException e) { 


l.setText("Can't open socket"); 


l.setText("Ready to send your email address"); 
iF 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(send)) { 
if(pl != null) { 
// pl.stop(); Deprecated in Java 1.2 


Thread remove = pl; 


pl = null; 
remove.interrupt(); 

} 

12.setText(""); 

// Check for errors in email name: 

str = t.getText().toLowerCase().trim(); 

if(str.indexOf(' ') != -1) { 
l.setText("Spaces not allowed in name"); 
return true; 

} 

if(str.indexOf(',') != -1) { 
l.setText("Commas not allowed in name"); 
return true; 

} 

if(str.indexOf('@') == -1) { 
l.setText("Name must include '@'"); 
12.setText(""); 
return true; 

} 

if(str.indexof('@') == 0) { 
1.setText("Name must preceed '@'"); 
12.setText(""); 


return true; 


} 
String end = 
str.substring(str.indexOf('@')); 
if(end.indexOf('.') == -1) { 
1l.setText("Portion after '@' must " + 
"have an extension, such as '.com'"); 
12.setText(""); 
return true; 
} 
// Everything's OK, so send the name. Get a 
// fresh buffer, so it's zeroed. For some 
// reason you must use a fixed size rather 
// than calculating the size dynamically: 
byte[] sbuf = 
new byte[NameCollector .BUFFER_SIZE]; 
str.getBytes(0, str.length(), sbuf, 0); 
DatagramPacket toSend = 
new DatagramPacket ( 
sbuf, 100, hostAddress, 
NameCollector.COLLECTOR_PORT); 
try { 
s.send(toSend); 


} catch(Exception e) { 


1l.setText("Couldn't send datagram"); 
return true; 
} 
l.setText("Sent: " + str); 
send.setLabel("Re-send"); 
pl = new Thread(this); 
pl.start(); 
12.setText( 
"Waiting for verification " + ++vcount); 
} 
else return super.action(evt, arg); 
return true; 

} 

// The thread portion of the applet watches for 
// the reply to come back from the server: 
public void run() { 

try { 
s.receive(dp); 

} catch(Exception e) { 
12.setText("Couldn't receive datagram"); 
return; 


} 


12.setText(new String(dp.getData(), 


©, 0, dp.getLength())); 
} 
} ///:~ 


程序 记 的 UI《〈 用 户 界面 ) 非常 简单 。 它 包含 了 一 个 TestField (CAS 
段 ) ， 以 便 我 们 键入 一 个 电子 函件 地 址 ， 以 及 一 个 Button〈 按 钮 ) ， 用 
于 将 地 址 发 给 服务 器 。 两 个 Label 〈 标 签 ) 用 于 向 用 户 报告 状态 信息 。 


到 现在 为 止 ， 大 家 已 能 判断 出 DatagramSocket、InetAddress、 组 冲 区 以 
及 DatagramPacket 都 属于 网 络 连 接 中 比较 矿 烦 的 部 分 。 最 后 ， 大 家 可 看 
J 分 ， 使 程序 片 能 够 “ 侦 听 ”由 服务 器 传 回 的 啊 应 


init(O) 方 法 用 大 家 熟悉 的 布局 工具 设置 GUI， 然 后 创建 DatagramSocket， 
它 将 同时 用 于 数据 报 的 收发 。 


action() 方 法 只 负责 监视 我 们 是 否 按 下 了 "发 送 ”(send) 按钮 。 记 住 ， 我 
们 已 被 限制 在 Java ”1.0 上 面 ， 所 以 不 能 再 用 较 灵 活 的 内 部 类 了 。 按 钮 按 
下 以 后 ， 采 取 的 第 一 项 行动 便 是 检查 线程 pl， 看 看 它 是 否 为 

null (42) 。 如 有 果 不 为 nuall， 表 明 有 一 个 活动 线程 正在 运行 。 消 息 首 次 发 
出 时 ， 会 局 动 一 个 新 线程 ， 用 和 它 监视 来 和 目 服 务 喜 的 回应 。 所 以 假若 有 个 
线程 正在 运行 ， 就 意味 着 这 并 非 用 户 第 一 次 发 送 消息 。pl 句 柄 被 设 为 
nul， 同 时 中 止 原 来 的 监视 者 〈 这 是 最 合理 的 一 种 做 法 ， 因 为 stopO 已 被 
Java 1.2“ 反 对 ”， 这 在 前 一 章 已 解释 过 了 ) 。 
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搜索 其 中 的 非法 字符 。 如 果 找 到 一 个 ， 就 把 情况 报告 给 用 户 。 注 意 进行 
所 有 这 些 工 作 时 ， 都 不 必 涉 及 网 络 通信 ， 所 以 速度 非常 快 ， 而 且 不 会 影 
啊 带 宽 和 服务 器 的 性 能 。 


名 字 校 验 通过 以 后 ， 它 会 打包 到 一 个 数据 报 里 ， 然 后 采用 与 前 面 那个 数 
据 报 示 例 一 样 的 方式 发 到 主机 地 址 和 端口 编号 。 第 一 个 标签 会 发 生变 
化 ， 指 出 已 成 功 发 送出 去 。 而 且 按 钮 上 的 文字 也 会 改变 ， 变 成 “ 重 

发 ”(resend) 。 这 时 会 启动 线程 ， 第 二 个 标签 则 会 告诉 我 们 程序 片 正 在 























等 候 来 自 服务 器 的 回应 。 


线程 的 run(0) 方 法 会 利用 NameSender 中 包含 的 DatagramSocket 来 接收 数据 
(receive()) ， 除 非 出 现 来 自 服务 器 的 数据 报 包 ， 否 则 receive(O) 会 暂时 处 
于 “堵塞 ?或 者 “暂停 ”状态 。 结 果 得 到 的 数据 包 会 放 进 NameSender 的 
DatagramPacketdp 中 。 数 据 会 从 包 中 提取 出 来 ， 并 置 入 NameSender 的 第 
二 个 标签 。 随 后 ， 线 程 的 执行 将 中 断 ， 成 为 一 个 “ 死 ? 线 程 。 徊 某 段 时 间 
里 没有 收 到 来 自 服 务 器 的 回应 ， 用 户 可 能 变 得 不 耐烦 ， 再 次 按 下 按钮 。 
这 样 做 会 中 断 当前 线程 〈 数 据 发 出 以 后 ， 会 再 建 一 个 新 的 ) 。 由 于 用 一 
个 线程 来 监视 回应 数据 ， 所 以 用 户 在 监视 期 间 仍 然 可 以 自由 使 用 UI。 


1. Web 页 


当然 ， 程 序 片 必 须 放 到 一 个 Web 页 里 。 下 面 列 出 完整 的 Web 页 源码 ， 稍 
微 研 究 一 下 就 可 看 出 ， 我 用 它 从 自己 开办 的 邮寄 列表 (Mailing List) 
里 自动 收集 名 字 。 


<HTML> 


<HEAD> 

<META CONTENT="text/html"> 

<TITLE> 

Add Yourself to Bruce Eckel's Java Mailing List 
</TITLE> 

</HEAD> 

<BODY LINK="#0000ff" VLINK="#800080" BGCOLOR="#ffffff"> 
<FONT SIZE=6><P> 

Add Yourself to Bruce Eckel's Java Mailing List 
</P></FONT> 


The applet on this page will automatically add your email add 
on Java Seminar” Multimedia CD. Type in your email address an 


<applet code=NameSender width=400 height=100> 

</applet> 

<HR> 

If after several tries, you do not get verification it means 
<A HREF="mailto:Bruce@EckelObjects.com"> 
Bruce@EckelObjects.com</A> 

</BODY> 


</HTML> 


程序 片 标记 〈<applet>) 的 使 用 非常 简单 ， 和 第 13 章 展示 的 那 一 个 并 没 
有 什么 区 别 。 


15.5.3 要 注意 的 问题 


前 面 采 取 的 似乎 是 一 种 完美 的 方法 。 没 有 CGI 编 程 ， 所 以 在 服务 器 启动 
一 个 CGI 程序 时 不 会 出 现 延 迟 。 数 据 报 方式 似乎 能 产生 非常 快 的 啊 应 。 

此 外 ， 一旦 Java 1.1 得 到 绝 大 多 数 人 的 采纳 ， 服 务 器 端的 那 一 部 分 束 可 
Be A nen ay 
HAD) 。 


但 必须 注意 到 一 些 问 题 。 其 中 一 个 特别 容易 忽略 : 由 于 Java 应 用 在 服务 
器 上 是 连续 运行 的 ， 而 且 会 把 大 多 数 时 间 花 在 Datagram.receive() 方 法 的 
等 候 上 和 面 ， 这 样 便 为 CPU 市 来 了 额外 的 开销 。 人 至 少 ， 我 在 自己 的 服务 器 
上 便 友 现 了 这 个 问题 。 男 一 方面 ， 那 个 服务 器 上 不 会 发 生 其 他 更 多 的 事 
情 。 而 且 假 如 我 们 使 用 一 个 任务 更 为 繁重 的 服务 器 ， 启 动 程序 

用 “nice”( 一 个 Unix 程 序 ， 用 于 防止 进程 贫 吃 CPU 资 源 ) 或 其 他 等 价 程 
序 即 可 解决 问题 。 在 许多 情况 下 ， 都 有 必要 留意 象 这 样 的 一 些 应 用 一 一 
一 个 堵塞 的 receive() 完 全 可 能 造成 CPU 的 次 痪 。 


第 二 个 问题 涉及 防火 墙 。 可 将 防火 增 理 解 成 自己 的 本 地 网 与 因特网 之 间 
的 一 道 墙 〈 实 际 是 一 个 专用 机 器 或 防火 墙 软 件 ) 。 它 监视 进出 因特网 的 
所 有 通信 ， 确 保 这 些 通信 不 违背 预 设 的 规则 。 

















防火 墙 显 得 多 少 有 些 保 守 ， 要 求 严 格 遵守 所 有 规则 。 假 如 没有 遵守 ， 它 
们 会 无 情 地 把 它们 拒 之 门 外 。 例 如 ， 假 设 我 们 位 于 防火 墙 后 面 的 一 个 网 
络 中 ， 开 始 用 Web 浏 览 器 同 因特网 连接 ， 防 火 墙 要 求 所 有 传输 都 用 可 以 
接受 的 http 端 口 同 服务 器 连接 ， 这 个 端口 是 80。 现 在 来 了 这 个 Java 程 序 

请 NameSender， 它 试图 将 一 个 数据 报 传 到 端口 8080， 这 是 为 了 越过 “ 受 
保护 ”的 端口 范围 0-1024 而 设置 的 。 防 火 增 很 目 然 地 把 它 想象 成 最 坏 的 

情况 一 一 有 人 使 用 病毒 或 者 非法 扫描 问 口 一 一 根本 不 允许 传输 的 继续 进 
行 。 


只 要 我 们 的 客户 建立 的 是 与 因特网 的 原始 连接 《比如 通过 典型 的 ISP 接 
Fenternet) ， 束 不 会 出 现 此 类 防火 墙 问题 。 但 也 可 能 有 一 些 重 要 的 客户 
隐藏 在 防火 墙 后， 他 们 便 不 能 使 用 我 们 设计 的 程序 。 


在 学 过 有 关 Java 的 这 么 多 东西 以 后 ， 这 是 一 件 使 人 相当 泪 表 的 事情 ， 
为 看 来 必须 放弃 在 服务 器 上 使 用 Java， 改 为 学 习 如 何 编 写 C 或 Perl 脚 本 程 
序 。 但 请 大 家 不 要 绝望 。 


一 个 出 色 方 案 是 由 Sun 公 司 提 出 的 。 如 一 切 按 计划 进行 ，Web 服 务 器 最 
终 都 装备 “小 服务 程序 ?或 者 “服务 程序 请 ”〈Servlet) 。 它 们 负责 接收 来 
自 客 户 的 请 求 〈 经 过 防火 墙 允许 的 80 端 口 ) 。 而 且 不 再 是 启动 一 个 CGI 
程序 ， 它 们 会 局 动 小 服务 程序 。 根 据 Sun 的 设想 ， 这 些小 服务 程序 都 是 
用 Java 编 写 的 ， 而 且 只 能 在 服务 器 上 运行 。 运 行 这 种 小 程序 的 服务 器 会 
目 动 局 动 它们 ， 令 其 对 客户 的 请 求 进行 处 理 。 这 意味 着 我 们 的 所 有 程序 
都 可 以 用 Java 写 成 100% 纯 咖啡 )。 这 显然 是 一 种 非常 吸引 人 的 想法 : 
一 旦 习惯 了 Java， 束 不 必 换 用 其 他 语言 在 服务 絮 上 人 处理 客 户 请 求 。 


由 于 只 能 在 服务 器 上 控制 请 求 ， 所 以 小 服务 程序 API 没 有 提供 GUI 功 
能 。 这 对 NameCollector.java 来 说 非常 适合 ， 它 本 来 就 不 需要 任何 图 形 界 
面 。 




















在 本 书写 作 时 ，java.sun.com 已 提供 了 一 个 非常 廉价 的 小 服务 程序 专用 
服务 器 。Sun 辟 励 其 他 Web 服 务 器 开发 者 为 他 们 的 服务 器 软件 产品 加 入 
对 小 服务 程序 的 文 持 。 


15.6 Java 与 CGI 的 沟通 


Java 程 序 可 向 一 个 服务 器 发 出 一 个 CGI 请 求 ， 这 与 HTML 表 单 页 没什么 

两 样 。 而 且 和 HTML 页 一 样 ， 这 个 请 求 既 可 以 设 为 GET 〈 下 载 ) ， 亦 可 
设 为 POST (CEE) 。 除 此 以 外 ，Java 程 序 还 可 拦截 CGI 程序 的 输出 ， 所 
以 不 必 依 赖 程序 来 格式 化 一 个 新 页 ， 也 不 必 在 出 错 的 时 候 强 迫 用 户 从 一 
a a 
二 致 。 


代码 也 要 简单 一 些 ， 毕 竟 用 CGI 也 不 是 很 难 就 能 写 出 来 〈 前 提 是 真正 地 
HC) 。 所 以 在 这 一 节 里 ， 我 们 准备 办 个 CGI 编程 速成 班 。 为 解决 常 
规 问 题 ， 将 用 C++ 创建 一 些 CGI 工 具 ， 以 便 我 们 编写 一 个 能 解决 所 有 问 
题 的 CGI 程序 。 这 样 做 的 好 处 是 移植 能 力 特 别 强 一 一 即将 看 到 的 例子 能 
在 文 持 CGI 的 任何 系统 上 运行 ， 而 且 不 存在 防火 墙 的 问题 。 


这 个 例子 也 前 示 了 如 何在 程序 片 CApplet) 和 CGI 程序 之 间 建 立 连 接 ， 
以 便 将 其 方便 地 改编 到 目 己 的 项 目 中 。 


15.6.1 CGI 数据 的 编码 


在 这 个 版 本 中 ， 我 们 将 收集 名 字 和 电子 函件 地 址 ， 并 用 下 述 形式 将 其 保 
FF BICEP: 














First Last <email@domain.com>; 


这 对 任何 E-mail 程序 来 说 部 是 一 种 非常 方便 的 格式 。 由 于 只 需 收 集 两 个 
字段 ， 而 且 CGI 为 字段 中 的 编码 采用 了 一 种 特殊 的 格式 ， 所 以 这 里 没有 
简便 的 方法 。 如 果 目 己 动 手 编制 一 个 原始 的 HTML 页 ， 并 加 入 下 述 代码 
行 ， 即 可 正确 地 理解 这 一 点 : 


<Form method="GET" ACTION="/cgi-bin/Listmgr2.exe"> 


<P>Name: <INPUT TYPE = "text" NAME = "name" 
VALUE = "" size = "40"></p> 


<P>Email Address: <INPUT TYPE = "text" 


NAME = "email" VALUE = "" size = "40"></p> 
<p><input type = "submit" name = "Submit" > </p> 


</Form> 


上 述 代 码 创 建 了 两 个 数据 输入 字段 CK) ， 名 为 name 和 email。 另 外 还 
有 一 个 submit( 提 交 ) 按钮 ， 用 于 收集 数据 ， 并 将 其 发 给 CGI 程序 。 
Listmgr2.exe 是 驻 留 在 特殊 程序 目录 中 的 一 个 可 执行 文件 。 在 我 们 的 Web 
服务 器 上 ， 该 目录 一 般 都 叫 作 “cgi-bin”( 注 释 @)) 。 如 果 在 那个 目录 里 
找 不 到 该 程序 ， 结 果 就 无 法 出 现 。 填 好 这 个 表单 ， 然 后 按 下 提交 按钮 ， 
即 可 在 浏览 器 的 URL 地 址 窗口 里 看 到 象 下 面 这 样 的 内 容 : 


http://www.myhome.com/cgi-bin/Listmgr2.exe? 
name=First+Last&email=email@domain.com&submit=Submit 


3): 在 Windows32 平 台 下 ， 可 利用 与 Microsoft Office 97 或 其 他 产品 配套 
提供 的 Microsoft Personal Web Server 〈 微 软 个 人 Web 服 务 器 ) 进行 测 
试 。 这 是 进行 试验 的 最 好 方法 ， 因 为 不 必 正 式 连 入 网 络 ， 可 在 本 地 环境 
中 完成 测试 (速度 也 非常 快 )。 如 果 使 用 的 是 不 同 的 平台 ， 或 者 没有 
Office 97 或 者 FrontPage 98 那 样 的 产品 ， 可 到 网 上 找 一 个 免费 的 Web 服 务 
att A omit. 


当然 ， 上 述 URL 实 际 显示 时 是 不 会 拆 行 的 。 从 中 可 稍微 看 出 如 何 对 数据 
编码 并 传 给 CGI。 至 少 有 一 件 事 情 能 够 肯定 一 一 空格 是 不 允许 的 (因为 
它 通常 用 于 分 隔 命 令 行 参数 ) 。 所 有 必需 的 空格 都 用 “+” 号 蔡 代 ， 每 个 
字段 都 包含 了 字段 名 (具体 由 HTML 页 决定 ) ， 后 面 跟随 一 个 “=” 号 以 
及 正式 的 字段 数据 ， 最 后 用 一 个 “&” 结 束 。 


到 这 时 ， 大 家 也 许 会 对 “+”，“=” 以 及 “&”* 的 使 用 产生 疑惑 。 假 如 必须 在 

字段 里 使 用 这 些 字 符 ， 那 么 该 如 何 声明 呢 ? 例如， 我 们 可 能 使 用 “John 

& MarshaSmith” 这 个 名 字 ， 其 中 的 “&*”* 代 表 “And”。 事 实 上 ， 它 会 编码 成 
下 面 这 个 样子 : 


John+%26+Marsha+Smith 


也 就 是 说 ， 特 殊 字 符 会 转换 成 一 个 “%”， 并 在 后 面 跟 上 它 的 十 六 进 制 
ASCII 编 码 。 














和 邓 运 的 是 ，Java 有 一 个 工具 来 帮助 我 们 进行 这 种 编码 。 这 是 URLEncoder 
类 的 一 个 静态 方法 ， 名 为 encode()。 可 用 下 述 程序 来 试验 这 个 方法 : 


//: EncodeDemo. java 





// Demonstration of URLEncoder.encode() 
import java.net.*; 
public class EncodeDemo { 
public static void main(String[] args) { 
String s = ""; 
for(int i = 0; i < args.length; i++) 
s += args[i] + " "; 
s = URLEncoder.encode(s.trim()); 
System.out.println(s); 
} 
} ///:~ 





该 程序 将 获取 一 些 命令 行 参数 ， 把 它们 合并 成 一 个 由 多 个 词 构成 的 字 
串 ， 各 词 之 间 用 空格 分 阳 (最 后 一 个 空格 用 String.trim() 吻 除了 〉。 随 后 
对 它们 进行 编码 ， 并 打印 出 来 。 


为 调用 一 个 CGI 程序 ， 程 序 片 要 做 的 全 部 事情 就 是 从 自己 的 字段 或 其 他 
地 方 收 集 数 据 ， 将 所 有 数据 都 编码 成 正确 的 URL 样 式 ， 然 后 汇编 到 单独 
一 个 字 串 里 。 每 个 字段 名 后 面 都 加 上 一 个 “=” 符 号 ， 紧 跟 正 式 数据 ， 再 
紧 跟 一 个 “&”。 为 构建 完整 的 CGI 命令 ， 我 们 将 这 个 字 串 置 于 CGI 程序 的 
URL 以 及 一 个 “?” 后 。 这 是 调用 所 有 CGI 程序 的 标准 方法 。 大 家 马上 就 会 
看 到 ， 用 一 个 程序 片 能 够 很 轻松 地 完成 所 有 这 些 编码 与 合并 。 


15.6.2 程序 片 














程序 片 实际 要 比 NameSender.java 简 单一 些 。 这 部 分 是 由 于 很 容易 即 可 发 
出 一 个 GET 请 求 。 此 外 ， 也 不 必 等 候 回复 信息 。 现 在 有 两 个 字段 ， 而 非 
一 个 ， 但 大 家 会 发 现 许多 程序 片 都 是 熟悉 的 ， 请 比较 NameSender.java。 


//: NameSender2.java 








// An applet that sends an email address 
// “ia a CGI GET, using Java 1.02. 
import java.awt.*; 
import java.applet.*; 
import java.net.*; 
import java.io.* 
public class NameSender2 extends Applet { 
final String CGIProgram = "Listmgr2.exe"; 
Button send = new Button( 
"Add email address to mailing list"); 
TextField name = new TextField( 
"type your name here", 40), 
email = new TextField( 
"type your email address here", 40); 
String str = new String(); 
Label 1 = new Label(), 12 = new Label(); 
int vcount = 0; 
public void init() { 


setLayout(new BorderLayout()); 


Panel p = new Panel(); 
p.setLayout(new GridLayout(3, 1)); 
p.add(name); 
p.add(email); 
p.add(send); 
add("North", p); 
Panel labels = new Panel(); 
labels.setLayout(new GridLayout(2, 1)); 
labels.add(1l); 
labels.add(12); 
add("Center", labels); 
l.setText("Ready to send email address"); 
} 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(send)) { 
12.setText(""); 
// Check for errors in data: 
if(name.getText().trim() 
.indexof(' ') == -1) { 
1.setText ( 
"Please give first and last name"); 
12.setText(""); 


return true; 


} 


str = email.getText().trim(); 
if(str.indexoOf(' ') != -1) { 
1.setText ( 
"Spaces not allowed in email name"); 
12.setText(""); 
return true; 
} 
if(str.indexof(',') != -1) { 
1.setText ( 
"Commas not allowed in email name"); 
return true; 
} 
if(str.indexOf('@') == -1) { 
l.setText("Email name must include '@'"); 
12.setText(""); 
return true; 
} 
if(str.indexOf('@') == 0) { 
1.setText ( 
"Name must preceed '@' in email name"); 
12.setText(""); 


return true; 


} 
String end = 
str.substring(str.indexOf('@')); 
if(end.indexOf('.') == -1) { 
1l.setText("Portion after '@' must " + 
"have an extension, such as '.com'"); 
12.setText(""); 
return true; 
} 
// Build and encode the email data: 
String emailData = 
"name=" + URLEncoder.encode( 
name.getText().trim()) + 
"®email=" + URLEncoder .encode( 
email.getText().trim().toLowerCase()) + 
"&submit=Submit"; 
// Send the name using CGI's GET process: 
try { 
1l.setText("Sending..."); 
URL u = new URL( 
getDocumentBase(), "cgi-bin/™" + 
CGIProgram + "?" + emailData); 


l.setText("Sent: " + email.getText()); 


send.setLabel("Re-send"); 
12.setText( 

"Waiting for reply " + ++vcount); 
DataInputStream server = 

new DataInputStream(u.openStream()); 
String line; 
while((line = server.readLine()) != null) 


12.setText(line); 


WY 


catch(MalformedURLException e) { 


1.setText("Bad URL"); 


WY 


catch(IOException e) { 


1l.setText("I0 Exception"); 


} 
else return super.action(evt, arg); 


return true; 


} 
} ///:~ 


CGI 程序 〈 不 久 即 可 看 到 ) 的 名 字 是 Listmgr2.exe。 许 多 Web 服 务 右 都 在 
Unix 机 右上 运行 (Linux 也 越 来 越 受 到 青睐 ) 。 根 据 传统 ， 它 们 一 般 不 
为 自己 的 可 执行 程序 采用 .exe 扩 展 名 。 但 在 Unix 操 作 系 统 中 ， 可 以 把 目 
己 的 程序 称呼 为 自己 希望 的 任何 东西 。 奎 使 用 的 是 .exe 扩 展 名 ， 程 序 考 
需 任 何 修改 即 可 通过 Unix 和 Win32 的 运行 测试 。 














和 往 第 一 样 ， 程 序 片 设置 了 自己 的 用 户 界 面 〈 这 次 是 两 个 输入 字段 ， 不 
是 一 个 ) 。 唯 一 显著 的 区 别 是 在 action() 方 法 内 产生 的 。 该 方法 的 作用 是 
对 按钮 按 下 事件 进行 控制 。 名 字 检 查 过 以 后 ， 大 家 会 及 现下 述 代 码 行 : 


String emailData = 


"name=" + URLEncoder .encode( 
name.getText().trim()) + 
"®email=" + URLEncoder.encode( 
email.getText().trim().toLowerCase()) + 
"&submit=Submit"; 
// Send the name using CGI's GET process: 
try { 
1.setText("Sending..."); 
URL u = new URL( 
getDocumentBase(), "Ccgi-bin/" + 
CGIProgram + "?" + emailData); 
l.setText("Sent: " + email.getText()); 
send.setLabel("Re-send"); 
12.setText( 
"Waiting for reply " + ++vcount); 
DataInputStream server = 
new DataInputStream(u.openStream()); 
String line; 


while((line = server.readLine()) != null) 


12.setText(line); 


LI caia 





name 和 email 数 据 都 是 它们 对 应 的 文字 框 里 提取 出 来 ， 而 且 两 端 多 余 的 
空格 都 用 trim0O 易 去 了 。 为 了 进入 列表 ，email 名 字 被 强制 换 成 小 写 形 
式 ， 以 便 能 够 准确 地 对 比 《〈 防 止 基于 大 小 写 形式 的 错误 判断 ) 。 来 自 每 
个 字段 的 数据 都 编码 为 URL 形 式 ， 随 后 采用 与 HIML 页 中 一 样 的 方式 汇 
编 GET 字 串 (这 样 一 来 ， 我 们 可 将 Java 程 序 片 与 现 有 的 任何 CGI 程 序 结 
合 使 用 ， 以 满足 常规 的 HTML GET 请 求 ) 。 


到 这 时 ， 一 些 Java 的 魔力 已 经 开始 发 挥 作用 了 : 如 果 想 同 任何 URL 连 
接 ， 只 需 创 建 一 个 URL 对 象 ， 并 将 地 址 传递 给 构建 器 即 可 。 构 建 器 会 负 
责 建 并 同 服务 器 的 连接 (对 Web 服 务 器 来 说 ， 所 有 连接 行动 都 是 根据 作 
为 URL 使 用 的 字 串 来 判断 的 ) 。 就 目前 这 种 情况 来 说，URL 指 辣 的 是 当 
前 Web 站 点 的 cgi-bin 目 录 (当前 Web 站 点 的 基础 地 址 是 用 
getDocumentBase() 设 定 的 ) 。 一 旦 Web 服 务 器 在 URL 中 看 到 了 一 个 “cgi- 
bin”， 会 接着 希望 在 它 后 面 跟随 了 cgi-bin 目 录 内 的 某 个 程序 的 名 字 ， 那 
是 我 们 要 运行 的 目标 程序 。 程 序 名 后 面 是 一 个 问号 以 及 CGI 程序 会 在 
QUERY_STRING 环 境 变 量 中 查找 的 一 个 参数 字 串 (马上 就 要 学 到 )。 


我 们 发 出 任何 形式 的 请 求 后 ， 一 般 都 会 得 到 一 个 回应 的 HTML 页 。 但 大 
使 用 Java 的 URL 对 象 ， 我 们 可 以 拦截 自 CGI 程 序 传 回 的 任何 东西 ， 只 需 
从 URL 对 象 里 取得 一 个 InputStream (输入 数据 流 ) 即 可 。 这 是 用 URL 对 
象 的 openStream() 方 法 实现 ， 它 要 封装 到 一 个 DataInputStream 里 。 随 后 

就 可 以 读 取 数 据 行 ， 若 readLine0 返 回 一 个 null〈 空 值 ) ， 就 表明 CGI 程 
序 已 结束 了 它 的 输出 。 


我 们 即将 看 到 的 CGI 程 序 返 回 的 仅仅 是 一 行 ， 它 是 用 于 标志 成 功 与 否 
《以 及 失败 的 具体 原因 ) 的 一 个 字 串 。 这 一 行 会 被 捕获 并 置 放 第 二 个 
Label 字 段 里 ， 使 用 户 看 到 具体 发 生 了 什么 事情 。 

1. 从 程序 片 里 显示 一 个 Web 页 


程序 亦 可 将 CGI 程序 的 结果 作为 一 个 web 页 显示 出 来 ， 就 象 它们 在 普通 
HTML 模 式 中 运行 那样 。 可 用 下 述 代 码 做 到 这 一 点 : 





























getAppletContext().showDocument(u); 


其 中 ，u 代 表 URL 对 象 。 这 是 将 我 们 重新 定 癌 于 为 一 个 Web 页 的 一 个 简 
单 例子 。 那 个 页 凑巧 是 一 个 CGI 程序 的 输出 ， 但 可 以 非常 方便 地 进入 一 
个 原始 的 HTML 页 ， 所 以 可 以 构建 这 个 程序 片 ， 令 其 产生 一 个 由 密码 保 
护 的 网 天 ， 通 过 它 进 入 自己 Web 站 点 的 特殊 部 分 : 


//: ShowHTML , java 


import java.awt.*; 
import java.applet.*; 
import java.net.*; 
import java.io.*; 
public class ShowHTML extends Applet { 
static final String CGIProgram = "MyCGIProgram"; 
Button send = new Button("Go"); 
Label 1 = new Label(); 
public void init() { 
add(send); 
add(1); 
} 
public boolean action (Event evt, Object arg) { 
if(evt.target.equals(send)) { 
try { 
// This could be an HTML page instead of 
// a CGI program. Notice that this CGI 


// program doesn't use arguments, but 


// you can add them in the usual way. 
URL u = new URL( 
getDocumentBase(), 
"cgi-bin/" + CGIProgram); 
// Display the output of the URL using 
// the Web browser, as an ordinary page: 
getAppletContext().showDocument(u); 
} catch(Exception e) { 


l.setText(e.toString()); 


} 
else return super.action(evt, arg); 


return true; 


} 
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15.6.3 用 C++ 写 的 CGI 程序 


经 过 前 面 的 学 习 ， 大 家 应 该 能 够 根据 例子 用 ANSI ”CC 为 自己 的 服务 器 写 
出 CGI 程序 。 之 所 以 选用 ANSI C， 是 因为 它 几乎 随处 可 见 ， 是 最 流行 的 
C 语 言 标准 。 当 然 ， 现 在 的 C++ 也 非常 流行 了 ， 特 别 是 采用 GNU C++ 编 
译 器 〈g++) 形式 的 那 一 些 (注释) 。 可 从 网 上 许多 地 方 免费 下 载 
g++， 而 且 可 选用 几乎 所 有 平台 的 版 本 (通常 与 Linux 那 样 的 操作 系统 配 
套 提 供 ， 且 已 预先 安装 好 ) 。 正 如 大 家 即将 看 到 的 那样 ， 从 CGI 程序 可 
获得 面 癌 对 象 程序 设计 的 许多 好 处 。 








D: GNU 的 全 称 是 “Gnu's Not ”Unix”。 这 最 早 是 由 “自由 软件 基金 
Z” (FSF) 负责 开发 的 一 个 项 目 ， 致 力 于 用 一 个 免费 的 版 本 取代 原 有 的 
Unix 操 作 系 统 。 现 在 的 Linux 似 乎 正在 做 前 人 没有 做 到 的 事情 。 但 GNU 
工具 在 Linux 的 开发 中 扮演 了 至 关 重 要 的 角色 。 事 实 上 ，Linux 的 整套 软 
件 包 附带 了 数量 非常 多 的 GNU 组 件 。 


为 避免 第 一 次 就 提出 过 多 的 新 概念 ， 这 个 程序 并 未 打算 成 为 一 

个 “ 纯 *C++ 程 序 ， 有 些 代码 是 用 普通 C 写 成 的 一 一 尽管 还 可 选用 C++ 的 一 
些 蔡 用 形式 。 但 这 并 不 是 个 突出 的 问题 ， 因 为 该 程序 用 C++ 制 作 最 大 的 
好 处 就 是 能 够 创建 类 。 在 解析 CGI 信息 的 时 候 ， 由 于 我 们 最 关心 的 是 字 
段 的 “名 称 / 值 ? 对 ， 所 以 要 用 一 个 类 (Pair) 来 代表 单个 名 称 / 值 对 ; 

男 一 个 类 (CGl_ vector) 则 将 CGI 字 串 目 动 解析 到 它 会 容纳 的 Pair 对 象 里 
ee sere ， 这 样 即 可 在 有 空 的 时 候 把 每 个 Pair OO) 都 取出 








这 个 程序 同时 也 非常 有 趣 ， 因 为 它 演示 了 C++ 与 Java 相 比 的 许多 优 缺 
点 。 大 家 会 看 到 一 些 相似 的 东西 ;比如 class 关 键 字 。 访 问 控制 使 用 的 是 
完全 相同 的 关键 字 public 和 private， 但 用 法 却 有 所 不 同 。 它 们 控制 的 是 
一 个 块 ， 而 非 单 个 方法 或 字段 (也 束 是 说 ， 如 果 指 定 private:， 后 续 的 每 
个 定义 都 具有 private 属 性 ， 直 到 我 们 再 指定 public: 为 止 ) 。 另 外 在 创建 
一 个 类 的 时 候 ， 所 有 定义 都 自动 默认 为 private。 


在 这 儿 使 用 C++ 的 一 个 原因 是 要 利用 C++“ 标 准 模板 库 ”(STL ) 提供 的 
人 便利。 至少，STL 包 含 了 一 个 vector 类 。 这 是 一 个 C++ 模板 ， 可 在 编译 期 
间 进 行 配置 ， 令 其 只 容纳 一 种 特定 类 型 的 对 象 〈 这 里 是 Pair 对 象 ) 。 和 
Java 的 Vector 不 同 ， 如 果 我 们 试图 将 除 Pair 对 象 之 外 的 任何 东西 置 入 
vector，C++ 的 vector 模 板 都 会 造成 一 个 编译 期 错误 ; 而 Java 的 Vector 能 
够 照 单 全 收 。 而 且 从 vector 里 取出 什么 东西 的 时 候 ， 它 会 目 动 成 为 一 个 
Pair 对 象 ， 毋 需 进 行 造型 处 理 。 所 以 检查 在 编译 期 进行 ， 这 使 程序 显得 
更 为 “健壮 ”。 此 外 ， 程 序 的 运行 速度 也 可 以 加 快 ， 因 为 没有 必要 进行 运 
行 期 间 的 造型 。vector 也 会 过 载 operator[]， 所 以 可 以 利用 非常 方便 的 语 
法 来 提取 Pair 对 象 。vector 模 板 将 在 CGI_vector 创 建 时 使 用 ;， 在 那 时 ， 大 
家 就 可 以 体会 到 如 此 简短 的 一 个 定义 居然 强 藏 有 那么 巨大 的 能 量 。 


右 提 到 缺 点 ， 就 一 定 不 要 瑟 记 Pair 在 下 列 代码 中 定义 时 的 复杂 程度 。 与 
我 们 在 Java 代 码 中 看 到 的 相 比 ，Pair 的 方法 定义 要 多 得 多 。 这 是 由 于 
C++ 的 程序 员 必 须 所 前 知道 如 何 用 副本 构建 如 控制 复制 过 程 ， 而 且 要 用 
过 载 的 operator= 完 成 赋值 。 正 如 第 12 半 解释 的 那样 ， 我 们 有 时 也 要 在 

















Tva 中 考虑 同样 的 事情 。 但 在 C++ 中 ， 几 乎 一刻 部 不 能 放松 对 这 些 站 是 
9 关注 。 


这 个 项 目 首先 创建 一 个 可 以 重复 使 用 的 部 分 ， 由 C++ 头 文 件 中 的 Pair 和 

CGI _vector 构 成。 从 技术 角度 看 ， 确 实 不 应 把 这 些 东西 都 塞 到 一 个 头 文 
件 里 。 但 就 目前 的 例子 来 说 ， 这 样 做 不 会 造成 任何 方面 的 损害 ， 而 且 更 
具有 Java 风 格 ， 所 以 大 家 阅读 理解 代码 时 要 显得 轻松 一 些 : 


//: CGITools.h 





// Automatically extracts and decodes data 
// from CGI GETs and POSTs. Tested with GNU C++ 
// (available for most server machines). 
#include <string.h> 
#include <vector> // STL vector 
using namespace std; 
// A class to hold a single name-value pair from 
// a CGI query. CGI_vector holds Pair objects and 
// returns them from its operator[]. 
class Pair { 

char* nm; 

char* val; 
public: 

Pair() { nm = val = 0; } 

Pair(char* name, char* value) { 


// Creates new memory: 


nm = decodeURLString(name) ; 
val = decodeURLString(value); 
} 
const char* name() const { return nm; } 
const char* value() const { return val; } 
// Test for "emptiness" 
bool empty() const { 
return (nm == 0) || (val == 0); 
} 
// Automatic type conversion for boolean test: 
operator bool() const { 
return (nm != 0) && (val != 0); 
} 
// The following constructors & destructor are 
// necessary for bookkeeping in C++. 
// Copy-constructor: 
Pair(const Pair& p) { 
if(p.nm == || p.val == 0) { 
nm = val = 0; 
} else { 
// Create storage & copy rhs values: 
nm = new char[strlen(p.nm) + 1]; 


strcpy(nm, p.nm); 


val = new char[strlen(p.val) + 1]; 


strcpy(val, p.val); 


i 


// Assignment operator: 
Pair& operator=(const Pair& p) { 
// Clean up old lvalues: 
delete nm; 
delete val; 
if(p.nm == || p.val == 0) { 
nm = val = 0; 
} else { 
// Create storage & copy rhs values: 
nm = new char[strlen(p.nm) + 1]; 
strcpy(nm, p.nm); 
val = new char[strlen(p.val) + 1]; 
strcpy(val, p.val); 
} 
return *this; 
} 
~Pair() { // Destructor 
delete nm; // © value OK 


delete val; 


} 


// If you use this method outide this class, 
// you're responsible for calling 'delete' on 
// the pointer that's returned: 
static char* 

decodeURLString(const char* URLstr) { 

int len = strlen(URLstr); 
char* result = new char[len + 1]; 
memset(result, len + 1, 0); 


for(int i= 0, j = 0; i <= len; i++, j++) { 


if(URLstr[i] == '+') 
result[j] = ' '; 

else if(URLstr[i] == '%') { 
result[j] = 


translateHex(URLstr[i + 1]) * 16 + 
translateHex(URLstr[i + 2]); 
1 += 2; // Move past hex code 
} else // An ordinary character 
result[j] = URLstr[i]; 


} 


return result; 


} 


// Translate a single hex character; used by 


// decodeURLString(): 
static char translateHex(char hex) { 
if(hex >= 'A') 
return (hex & Oxdf) - 'A' + 10; 
else 

return hex - 'O'; 

小 
}; 
// Parses any CGI query and turns it 
// into an STL vector of Pair objects: 
class CGI_vector : public vector<Pair> { 

char* qry; 

const char* start; // Save starting position 
// Prevent assignment and copy-construction: 
void operator=(CGI_vector&); 

CGI_vector(CGI_vector&); 
public: 

// const fields must be initialized in the C++ 
// "Constructor initializer list": 

CGI_vector(char* query) 

start(new char[strlen(query) + 1]) { 


dry = (char*)start; // Cast to non-const 


strcpy(qry, query); 


Pair p; 
while((p = nextPair()) != 0) 
push_back(p); 

} 

// Destructor: 

~CGI_vector() { delete start; } 
private: 

// Produces name-value pairs from the query 
// string. Returns an empty Pair when there's 
// no more query string left: 

Pair nextPair() { 

char* name = qry; 
if(name == || *name == '\O') 
return Pair(); // End, return null Pair 
char* value = strchr(name, '='); 
if(value == 0) 
return Pair(); // Error, return null Pair 
// Null-terminate name, move value to start 
// of its set of characters: 
*value = '\O'; 
value++; 
// Look for end of value, marked by '&': 


qry = strchr(value, '&'); 


if(qry == 0) gry = ""; // Last pair found 
else { 
*qry = '\O'; // Terminate value string 
qryt++; // Move to next pair 
} 
return Pair(name, value); 
} 
}; ///:~ 





在 #include 语 句 后 ， 可 看 到 有 一 行 是 : 
using namespace std; 


C++ 中 的 “命名 空间 ”(Namespace) 解决 了 由 Java 的 package 负 责 的 一 个 
问题 : 将 库 名 隐藏 起 来 。std 命 名 空间 引用 的 是 标准 C++ 库 ， 而 vector 束 
在 这 个 库 中 ， 所 以 这 一 行 是 必需 的 。 


Pair 类 表面 看 异常 简单 ， 只 是 容纳 了 两 个 (private〉 字符 指针 而 已 一 一 
一 个 用 于 名 字 ， 另 一 个 用 于 值 。 默 认 构 建 器 将 这 两 个 指针 人 简单 地 设 为 
零 。 这 是 由 于 在 C++ 中 ， 对 象 的 内 存 不 会 自动 置 零 。 第 二 个 构建 句 调 用 
方法 decodeURLString0， 在 新 分 配 的 推 内存 中 生成 一 个 解码 过 后 的 字 
串 。 这 个 内 存 区 域 必须 由 对 象 负责 管 理 及 清除 ， 这 与 “破坏 器 ?中 见 到 的 
相同 。name() 和 value() 方 法 为 相关 的 字段 产生 只 读 指 针 。 利 用 empty0 方 
法 ， 我 们 查询 Pair 对 象 它 的 某 个 字段 是 否 为 空 ， 返 回 的 结果 是 一 个 bool 
C++ 内 建 的 基本 布尔 数据 类 型 。operator bool0 使 用 的 是 C++“ 运 算 符 
过 载 ” 的 一 种 特殊 形式 。 它 允许 我 们 控制 自动 类 型 转换 。 如 果 有 一 个 名 
为 p 的 Pair 对 象 ， 而 且 在 一 个 本 来 希望 是 布尔 结果 的 表达 式 中 使 用 ， 比 如 
if(p){/...， 那 么 编译 器 能 辨别 出 它 有 一 个 Pair， 而 且 需 要 的 是 个 布尔 值 ， 
所 以 自动 调用 operator bool()， 进 行 必 要 的 转换 。 


接 下 来 的 三 个 方法 属于 第 规 编码 ， 在 C++ 中 创建 类 时 必须 用 到 它们 。 根 
气 C++ 类 采用 的 所 谓 “ 经 典 形 式 ”， 我 们 必须 定义 必要 的 “原始 ?构建 融 ， 
































以 及 一 个 副本 构建 器 和 赋值 运算 符 一 operator=〈 以 及 破坏 器 ， 用 于 清 
除 内 存 〉。 之 所 以 要 作 这 样 的 定义 ， 是 由 于 编译 器 会 “默默 ”地 调用 它 
们 。 在 对 象 传 入 、 传 出 一 个 函数 的 时 候 ， 需 要 调用 副本 构建 器 ;而 在 分 
配对 象 时 ， 需 要 调用 赋值 运算 符 。 只 有 真正 掌握 了 副本 构建 右 和 赋值 运 
算 符 的 工作 原理 ， 才 能 在 C++ 里 写 出 真正 “健壮 ”的 类 ， 但 这 需要 需要 一 
个 比较 艰苦 的 过 程 GERO) à- 


©: 我 的 《Thinking in C++) (Prentice-Hall,1995) 用 了 一 整 章 的 地 方 
来 讨论 这 个 主题 。 知 需 更 多 的 帮助 ， 请 务必 看 看 那 一 章 。 


只 要 将 一 个 对 象 按 值 传 入 或 传 出 函数 ， 束 会 自动 调用 副本 构建 右 
Pair(const Pair&s)。 也 就 是 说 ， 对 于 准备 为 其 制作 一 个 完整 副本 的 那个 对 
象 ， 我 们 不 准备 在 函数 框架 中 传递 它 的 地 址 。 这 并 不 是 Java 提 供 的 一 个 
选项 ， 由 于 我 们 只 能 传递 句柄 ， 所 以 在 Java 里 没有 所 谓 的 副本 构建 喜 
(如 果 想 制作 一 个 本 地 副本 ， 可 以 “克隆 ”那个 对 象 一 一 使 用 clone()， 参 
见 第 12 章 ) 。 类 似 地 ， 如 果 在 Java 里 分 配 一 个 句柄 ， 它 会 简单 地 复制 。 
但 C++ 中 的 赋值 意味 着 整个 对 象 都 会 复制 。 在 副本 构建 嚣 中， 我 们 创建 
新 的 存储 空间 ， 并 复制 原始 数据 。 但 对 于 赋值 运算 符 ， 我 们 必须 在 分 配 
新 存储 空间 之 前 释放 老 存 储 空 间 。 我 们 要 见 到 的 也 许 是 C++ 类 最 复杂 的 
一 种 情况 ， 但 那 正 是 Java 的 支持 者 们 论证 Java 比 C++ 简 单 得 多 的 有 力 证 
据 。 在 Java 中 ， 我 们 可 以 自由 传递 句柄 ， 善 后 工作 则 由 垃圾 收集 器 负 
贡 ， 所 以 可 以 轻松 许多 。 

但 事情 并 没有 完 。Pair 类 为 nm 和 val 使 用 的 是 char*， 最 复杂 的 情况 主要 
是 围绕 指针 展开 的 。 如 果 用 较 时 侣 的 C++ string KRR char, 
要 变 得 简单 得 多 (当然 ， 并 不 是 所 有 编译 占 都 提供 了 对 string 的 支 

FF) 。 那 么 ，Pair 的 第 一 部 分 看 起 来 就 象 下 面 这 样 : 


class Pair { 
































string nm; 

string val; 
public: 

Pair() { } 


Pair(char* name, char* value) { 


nm = decodeURLString(name) ; 

val = decodeURLString(value); 
} 
const char* name() const { return nm.c_str(); } 
const char* value() const { 

return val.c_str(); 
} 
// Test for "emptiness" 
bool empty() const { 

return (nm.length() == 0) 

|| (val.length() == 0); 

} 
// Automatic type conversion for boolean test: 
operator bool() const { 

return (nm.length() != 0) 


&& (val.length() != 0); 


(此 外 ， 对 这 个 类 decodeURLString0 会 返回 一 个 string， 而 不 是 一 个 
char*) 。 我 们 不 必定 义 副本 构建 嚣 、operator= 或 者 破坏 器 ， 因 为 编译 器 
己 帮 我 们 做 了 ， 而 且 做 得 非常 好 。 但 即使 有 些 事情 是 自动 进行 的 ， 
C++ 程序 员 也 必须 了 解 副 本 构建 以 及 赋值 的 细节 。 


Pair 类 剩 下 的 部 分 由 两 个 方法 构成 : decodeURLString0 以 及 一 个 “帮助 
器 ”方法 translateHex() 一 一 将 由 decodeURLString() 使 用 。 注 意 
translateHex() 并 不 能 防范 用 户 的 恶意 输入 ， 比 如 “%1H”。 分 配 好 足够 的 





存储 空间 后 (必须 由 破坏 嚣 释放) ，decodeURLString() 就 会 其 中 遍历 ， 
将 所 有 “+? 都 换 成 一 个 空格 ;将 所 有 十 六 进 制 代码 《〈 以 一 个 “%" 打 头 ) 换 
成 对 应 的 字符 。 


CGI_vector 用 于 解析 和 容纳 整个 CGI GET 命 令 。 它 是 从 STL vector 里 继承 
的 ， 后 者 例 示 为 容纳 Pair。C++ 中 的 继承 是 用 一 个 冒号 表示 ， 在 Java 中 则 
要 用 extends。 上 此外， 继承 默认 为 private 属 性 ， 所 以 几乎 肯定 需要 用 到 
public 关 键 字 ， 束 象 这 样 做 的 那样 。 大 家 也 会 发 现 CGI_vector 有 一 个 副 
本 构建 器 以 及 一 个 operator=， 但 它们 都 声明 成 private。 这 样 做 是 为 了 防 
止 编 译 器 同步 两 个 函数 (如 果 不 上 自己 声明 它们 ， 两 者 束 会 同步 ) 。 但 这 
同时 也 禁止 了 客户 程序 员 按 值 或 者 通过 赋值 传递 一 个 CGI_vector。 


CGI_vector 的 工作 是 获取 QUERY_STRING， 并 把 它 解 析 成 “名 称 / 
值 ? 对 ， 这 需要 在 Pair 的 帮助 下 完成 。 它 首先 将 字 串 复制 到 本 地 分 配 的 内 
存 ， 并 用 常数 指针 start 跟 踪 起 始 地 址 〈 稍 后 会 在 破坏 器 中 用 于 释放 内 
F) 。 随 后 ， 它 用 自己 的 nextPair() 方 法 将 字 串 解析 成 原始 的 “名 称 / 
值 ? 对 ， 各 个 对 之 间 用 一 个 “=” 利 “&” 符 号 分 隔 。 这 些 对 由 nextPair0O 传 递 
给 Pair 构 建 器 ， 所 以 nextPair() 返 回 的 是 一 个 Pair 对 象 。 随 后 用 push_back0) 
将 该 对 象 加 入 vector。nextPair() 裔 历 完整 个 QUERY_STRING 后 ， 会 返回 
一 个 零 值 。 


现在 基本 工具 已 定义 好 ， 它 们 可 以 简 蛙 地 在 一 个 CGI 程 序 中 使 用 ， 就 象 
下 面 这 样 : 


//: Listmgr2.cpp 


























// CGI version of Listmgr.c in C++, which 

// extracts its input via the GET submission 
// from the associated applet. Also works as 
// an ordinary CGI program with HTML forms. 
#include <stdio.h> 

#include "CGITools.h" 


const char* dataFile = "list2.txt"; 


const char* notify = "Bruce@EckelObjects.com"; 
#undef DEBUG 
// Similar code as before, except that it looks 
// for the email name inside of '<>': 
int inList(FILE* list, const char* emailName) { 
const int BSIZE = 255; 
char lbuf[BSIZE]; 
char emname[BSIZE]; 
// Put the email name in '<>' so there's no 
// possibility of a match within another name: 
sprintf(emname, "<%s>", emailName); 
// Go to the beginning of the list: 
fseek(list, ©, SEEK_SET); 
// Read each line in the list: 
while(fgets(lbuf, BSIZE, list)) { 
// Strip off the newline: 
char * newline = strchr(lbuf, '\n'); 
if(newline != 0) 
*newline = '\O'; 
if(strstr(lbuf, emname) != 0) 
return 1; 


} 


return 0; 


} 
void main() { 
// You MUST print this out, otherwise the 
// server will not send the response: 
printf("Content-type: text/plain\n\n"); 
FILE* list = fopen(dataFile, "a+t"); 
if(list == 0) { 
printf("error: could not open database. "); 
printf("Notify %s", notify); 
return; 
} 
// For a CGI "GET," the server puts the data 
// in the environment variable QUERY_STRING: 
CGI_vector query(getenv("QUERY_STRING")); 
#if defined(DEBUG) 
// Test: dump all names and values 
for(int i = 0; i < query.size(); i++) { 
printf("query[%d].name() = [%s], ", 
i, query[i].name()); 
printf("query[%d].value() = [%s]\n", 
i, query[i].value()); 
} 


#endif (DEBUG) 


Pair name = query[0]; 

Pair email = query[1]; 

if(name.empty() || email.empty()) { 
printf("error: null name or email"); 
return; 

} 

if(inList(list, email.value())) { 
printf("Already in list: %s", email.value()); 
return; 

} 

// It's not in the list, add it: 

fseek(list, ©, SEEK_END); 

fprintf(list, "%s <%s>;\n", 
name.value(), email.value()); 

fflush(list); 

fclose(list); 

printf("%s <%s> added to list\n", 
name.value(), email.value()); 


二 Pl ie 





alreadyInListO 函 数 与 前 一 个 版 本 几乎 是 完全 相同 的 ， 只 是 它 假定 所 有 电 
子 函件 地 址 都 在 一 个 “<>” 内 。 


在 使 用 GET 方 法 时 (通过 在 FORM3 引 导 命 令 的 METHOD 标 记 内 部 设置 ， 
但 这 在 这 里 由 数据 发 送 的 方式 控制 ) ，Web 服 务 器 会 收集 位 于 “?” 后 面 





的 所 有 信息 ， 并 把 它们 置 入 环境 变量 QUERY_STRING 〈 和 碍 询 字 串 ) 

里 。 所 以 为 了 读 取 那些 信息 ， 必 须 获得 QUERY_STRING 的 值 ， 这 是 用 
标准 的 C 库 函数 getnv0 完 成 的 。 在 main0 中 ， 注 意 对 QUERY_STRING 的 
解析 有 和 多么 容易 :， 只 需 把 它 传递 给 用 于 CGI _ vector 对 象 的 构建 器 《〈 名 为 
query) ， 剩 下 的 所 有 工作 都 会 自动 进行 。 从 这 时 开始 ， 我 们 束 可 以 从 
query 中 取出 名 称 和 值 ， 把 它们 当 作 数组 看 待 〈 这 是 由 于 operator[] 在 
vector 里 已 经 过 载 了 ) 。 在 调试 代码 中 ， 大 家 可 看 到 这 一 切 是 如 何 运作 
的 ;调试 代码 封装 在 预 处 理 器 引导 命令 术 f defined(DEBUG) 和 
#endif(DEBUG) 之 间 。 


现在 ， 我 们 迫切 需要 掌握 一 些 与 CGI 有 关 的 东西 。CGI 程 序 用 两 个 方式 
之 一 传递 它们 的 输入 : 在 GET 执 行 期 间 通 过 QUERY_STRING 传 递 〈 目 
前 用 的 这 种 方式 ) ， 或 者 在 POST 期 间 通 过 标准 输入 。 但 CGI 程序 通过 标 
准 输 出 及 送 自 己 的 输出 ， 这 通常 是 用 C 程 序 的 printf0 命 令 实现 的 。 那 么 
这 个 输出 到 哪里 去 了 呢 ? 它 回 到 了 Web 服 务 器 ， 由 服务 器 决定 该 如 何 处 
理 它 。 服 务 器 作出 决定 的 依据 是 contentrtype〈 内 容 类 型 ) 头 数据 。 这 意 
味 着 假如 content-type 头 不 是 它 看 到 的 第 一 件 东西 ， 就 不 知道 该 如 何 处 理 
因此 ， 我 们 无 论 如 何 也 要 使 所 有 CGI 程 序 都 从 content-type 
头 开 始 输出 。 


在 目前 这 种 情况 下 ， 我 们 希望 服务 器 将 所 有 信息 都 直接 反馈 回 客户 程序 

《 亦 即 我 们 的 程序 片 ， 它 们 正在 等 候 给 自己 的 回复 ，。 信 息 应 该 原封 不 
动 ， 所 以 content-type 设 为 text/plain( 纯 文本 ) 。 一 旦 服务 器 看 到 这 个 
头 ， 束 会 将 所 有 字 串 都 直接 发 还 给 客户 。 所 以 每 个 字 串 (三 个 用 于 出 错 
条 件 ， 一 个 用 于 成 功 的 加 入 ) 都 会 返回 程序 片 。 


我 们 用 相同 的 代码 添加 电子 函件 名 称 《〈“ 用 户 的 姓名 ) 。 但 在 CGI 脚本 的 
情况 下 ， 并 不 存在 无 限 循环 一 一 程序 只 是 简单 地 啊 应 ， 然 后 就 中 断 。 每 
次 有 一 个 CGI 请 求 抵达 时 ， 程 序 都 会 月 动 ， 对 那个 请 求 作 出 反应 ， 然 后 
自行 天 闭 。 所 以 CPU 不 可 能 陷入 空 等 等 的 尴 罚 境地， 只 有 局 动 程 序 和 打 
开 文 件 时 才 存 在 性 能 上 的 隐患 。Web 服 务 器 对 CGI 请 求 进行 控制 时 ， 它 
的 开销 会 将 这 种 隐患 减轻 到 最 低 程 度 。 


这 种 设计 的 另 一 个 好 处 是 由 于 Pair 和 CGI_vector 都 得 到 了 定义 ， 大 多 数 
工作 都 帮 我 们 目 动 完成 了 ， 所 以 只 需 修 改 main0 即 可 轻松 创建 自己 的 
CGI 程序 。 尽 管 小 服务 程序 (Servlet) 最 终 会 变 得 越 来 越 流行 ， 但 为 了 
创建 快速 的 CGI 程序 ，C++ 仍 然 显 得 非常 方便 。 














15.6.4 POST 的 概念 


在 许多 应 用 程序 中 使 用 GET 都 没有 问题 。 但 是 ，GET 要 求 通过 一 个 环境 
变量 将 自己 的 数据 传递 给 CGI 程序 。 但 假如 GET 字 串 过 长 ， 有 些 Web 服 
务 右 可 能 用 光 目 己 的 环境 空间 〈( 辱 字 串 长 度 超 过 200 字 符 ， 就 应 开始 关 
心 这 方面 的 问题 ) 。CGI 为 此 提供 了 一 个 解决 方案 : POST。 通 过 
POST， 数 据 可 以 编码 ， 并 按 与 GET 相 同 的 方法 连结 起 来 。 但 POST 利用 
标准 输入 将 编码 过 后 的 查询 字 串 传递 给 CGI 程序 。 我 们 要 做 的 全 部 事情 
就 是 判断 查询 字 串 的 长 度 ， 而 这 个 长 度 已 在 环境 变量 
CONTENT_LENGTH 中 保存 好 了 。 一 旦 知道 了 长 度 ， 就 可 自由 分 配 存 储 
空间 ， 并 从 标准 输入 中 读 入 指定 数量 的 字符 。 


对 一 个 用 来 控制 POST 的 CGI 程序 ， 由 CGITools.h 提 供 的 Pair 和 
CGI_vector 均 可 不 加 丝毫 改变 地 使 用 。 下 面 这 段 程序 揭示 了 写 这 样 的 一 
个 CGI 程序 有 多 么 简单 。 这 个 例子 将 采用 “ 纯 "C++， 所 以 studio.h 库 被 
iostream (IO 数据 流 ) 代替 。 对 于 iostream， 我 们 可 以 使 用 两 个 预先 定义 
好 的 对 象 : cin， 用 于 同 标准 输入 连接 ， 以 及 cout， 用 于 同 标准 输出 连 
接 。 有 几 个 办 法 可 从 cin 中 读 入 数据 以 及 癌 cout 中 写 入 。 但 下 面 这 个 程序 
准备 采用 标准 方法 : 用 “<<” 将 信息 发 给 cout， 并 用 一 个 成 员 函 数 〈 此 时 
是 read0) 从 cin 中 读 入 数据 : 


//: POSTtest ,cpp 








// CGI_vector works as easily with POST as it 

// does with GET. Written in "pure" C++. 

#include <iostream.h> 

#include "CGITools.h" 

void main() { 
cout << "Content-type: text/plain\n" << endl; 
// For a CGI "POST," the server puts the length 

// of the content string in the environment 


// variable CONTENT_LENGTH: 


char* clen = getenv("CONTENT_LENGTH"); 
if(clen == 0) { 
cout << "Zero CONTENT_LENGTH" << endl; 
return; 
} 
int len = atoi(clen); 
char* query_str = new char[len + 1]; 
cin.read(query_str, len); 
query_str[len] = '\0'; 
CGI_vector query(query_str); 
// Test: dump all names and values 
for(int i = 0; i < query.size(); i++) 
cout << "query[" << i << "].name() = [" << 
query[i].name() << "], " << 
"query[" << i << "].value() = [" << 
query[i].value() << "]" << endl; 
delete query_str; // Release storage 


} ///:~ 


getenv() K ŽORE 1a] —7 EB PREP, ABS ACER TEANGA RKE. 
若 指针 为 零 ， 表 明 CONTENT_LENGTH 环 境 变量 尚未 设置 ， 所 以 肯定 某 
个 地 方 出 了 问题 。 否 则 就 必须 用 ANSI “C 库 函数 atoi0 将 字 串 转换 成 一 个 
整数 。 这 个 长 度 将 与 new 一 起 运用 ， 分 配 足 够 的 存储 空间 ， 以 便 容 纳 查 
询 字 串 〈 男 加 它 的 空中 止 符 ) 。 随 后 为 cin() 调 用 read()。read0) 函 数 需 要 
取得 指向 目标 绥 冲 区 的 一 个 指针 以 及 要 读 入 的 字 节 数 。 随 后 用 空 字符 





(null〉 中 止 query_str， 指 出 已 经 抵达 字 串 的 末尾 ， 这 就 叫 作 “ 空 
iE” 


到 这 个 时 候 ， 我 们 得 到 的 查询 字 串 与 GET 查 询 字 串 已 经 没有 什么 区 别 ， 
所 以 把 它 传 递 给 用 于 CGI_vector 的 构建 器 。 随 后 便 和 前 例 一 样 ， 我 们 可 
以 自由 vector 内 不 同 的 字段 。 


为 测试 这 个 程序 ， 必 须 把 它 编 译 到 主机 Web 服 务 占 的 cgi-bin 目 录 下 。 然 
后 就 可 以 写 一 个 简单 的 HTML 页 进行 测试 ， 就 象 下 面 这 样 : 


<HTML> 


<HEAD> 

<META CONTENT="text/html"> 

<TITLE>A test of standard HTML POST</TITLE> 
</HEAD> 

Test, uses standard html POST 

<Form method="POST" ACTION="/cgi-bin/POSTtest"> 
<P>Fieldi1: <INPUT TYPE = "text" NAME = "Fieldi" 
VALUE = "" size = "40"></p> 

<P>Field2: <INPUT TYPE = "text" NAME = "Field2" 
VALUE = "" size = "4Q"></p> 

<P>Field3: <INPUT TYPE = "text" NAME = "Field3" 
VALUE = "" size = "40"></p> 

<P>Field4: <INPUT TYPE = "text" NAME = "Field4" 
VALUE = "" size = "40"></p> 


<P>Field5: <INPUT TYPE = "text" NAME = "Field5" 


VALUE = "" size = "40"></p> 


<P>Field6: <INPUT TYPE = "text" NAME = "Field6" 


VALUE = "" size = "40Q"></p> 

<p><input type = "submit" name = "submit" > </p> 
</Form> 

</HTML> 





填 好 这 个 表单 并 提交 出 去 以 后 ， 会 得 到 一 个 简单 的 文本 页 ， 其 中 包含 了 
解析 出 来 的 结果 。 从 中 可 知道 CGI 程序 是 否 在 正常 工作 。 


当然 ， 用 一 个 程序 片 来 提交 数据 显得 更 有 趣 一 些 。 然 而 ，POST 数 据 的 
提交 属于 一 个 不 同 的 过 程 。 在 用 常规 方式 调用 了 CGI 程 序 以 后 ， 必 须 为 
行 建立 与 服务 器 的 一 个 连接 ， 以 便 将 查询 字 串 反 馈 给 它 。 服 务 器 随后 会 
进行 一 番 处 理 ， 再 通过 标准 输入 将 查询 字 串 反馈 回 CGI 程 序 。 


为 建立 与 服务 器 的 一 个 直接 连接 ， 必 须 取得 自己 创建 的 URL， 然 后 调用 
openConnection() 创 建 一 个 URLConnection。 但 是 ， 由 于 URLConnection 
一 般 不 允许 我 们 把 数据 发 给 它 ， 所 以 必须 很 可 笑 地 调用 
setDoOutput(true) 函 数 ， 同 时 调用 的 还 包括 setDoInput(true) 以 及 
setAllowUserInteraction(false) 一 一 注释 @)。 最 后 ， 可 调用 
getOutputStream( 来 创建 一 个 OutputStream (输出 数据 流 ) ， 并 把 它 封 装 
到 一 个 DataOutputStream 里 ， 以 便 能 按 传 统 方式 同 它 通 信 。 下 面 列 出 的 
便 是 一 个 用 于 完成 上 述 工作 的 程序 片 ， 必 须 在 从 它 的 各 个 字段 里 收集 了 
数据 之 后 再 执行 它 : 


//: POSTtest ,java 


























// An applet that sends its data via a CGI POST 
import java.awt.*; 
import java.applet.*; 


import java.net.*; 


import java.io.*; 
public class POSTtest extends Applet { 
final static int SIZE = 10; 
Button submit = new Button("Submit"); 
TextField[] t = new TextField[SIZE]; 
String query = ""; 
Label 1 = new Label(); 
TextArea ta = new TextArea(15, 60); 
public void init() { 
Panel p = new Panel(); 
p.setLayout(new GridLayout(t.length + 2, 2)); 
for(int i = 0; i < t.length; i++) { 
p.add(new Label ( 
"Field "+ i+" ", Label.RIGHT)); 
p.add(t[i] = new TextField(30) ); 
} 
p.add(1); 
p.add(submit ) ; 
add("North", p); 
add("South", ta); 
} 
public boolean action (Event evt, Object arg) { 


if(evt.target.equals(submit)) { 


query = ""; 
ta.setText(""); 
// Encode the query from the field data: 
for(int i = 0; i < t.length; i++) 
query += "Field" + i+ "=" + 
URLEncoder . encode( 
t[i].getText().trim()) + 
"g": 
query += "submit=Submit"; 
// Send the name using CGI's POST process: 
try { 
URL u = new URL( 
getDocumentBase(), "cgi-bin/POSTtest"); 
URLConnection urlc = u.openConnection(); 
urlc.setDoOutput(true); 
urlc.setDoInput(true); 
urlc.setAllowUserInteraction(false); 
DataOutputStream server = 
new DataOutputStream( 
urlc.getOutputStream()); 
// Send the data 
server .writeBytes(query); 


server .close(); 


// Read and display the response. You 
// cannot use 
// getAppletContext().showDocument(u); 
// to display the results as a Web page! 
DataInputStream in = 
new DataInputStream( 
urlc.getInputStream()); 
String s; 
while((s = in.readLine()) != null) { 
ta.appendText(s + "\n"); 
} 
in.close(); 
} 
catch (Exception e) { 


l.setText(e.toString()); 


} 
else return super.action(evt, arg); 
return true; 


} 
Sdn 





我 不 得 不 说 自己 并 没有 真正 理解 这 儿 都 发 生 了 什么 事情 ， 这 些 概念 


都 是 从 Elliotte Rusty Harold 编 著 的 《Java Network Programming》 里 得 来 
的 ， 该 书 由 O'Reilly 于 1997 年 出 版 。 他 在 书 中 提 到 了 Java 连 网 函数 库 中 出 
现 的 许多 令 人 迷惑 的 Bug。 所 以 一 旦 涉足 这 些 领域 ， 事情 就 不 是 编写 代 

码 ， 然 后 让 它 自己 运行 那么 简单 。 一 定 要 警惕 潜在 的 陷阱 ! 





言 息 发 送 到 服务 器 后 ， 我 们 调用 getInputStream()， 并 把 返回 值 封装 到 一 
个 DataInputStream 里 ， 以 便 自己 能 读 取 结果 。 要 注意 的 一 件 事情 是 结果 
以 文本 行 的 形式 显示 在 一 个 TextArea (文本 区 域 》 中 。 为 什么 不 简单 地 
使 用 getAppletContext().showDocument(u) 呢 ?事实 上 ， 这 正 是 那些 陷阱 
中 的 一 个 。 上 述 代码 可 以 很 好 地 工作 ， 但 假如 试图 换 用 
showDocument()， 几 平一 切 都 会 停止 运行 。 也 束 是 说 ，showDocument() 
确实 可 以 运行 ， 但 从 POSTtest 得 到 的 返回 结果 是 “Zero 
CONTENT_LENGTH” (NAKA) 。 所 以 不 知道 为 什么 原因 ， 
showDocumentO 阻 止 了 POST 查询 向 CGI 程序 的 传递 。 我 很 难 判断 这 到 底 
是 一 个 在 以 后 版 本 里 会 修复 的 Bug， 还 是 由 于 我 的 理解 不 够 〈 我 看 过 的 
书 对 此 讲 得 都 很 模糊 ) 。 但 无 论 在 哪 种 情况 下 ， 只 要 能 坚持 在 文本 区 域 
里 观看 自 CGI 程 序 返 回 的 内 容 ， 上 述 程序 片 运行 时 就 没有 问题 。 


























15.7 用 JDBC 连 接 数 据 库 


据 估 算 ， 将 近 一 半 的 软件 开发 都 要 涉及 客户 〈 机 ) /服务 吉方 面 的 操 
作 。Java 为 自己 保证 的 一 项 出 色 能 力 就 是 构建 与 平台 无 关 的 客户 机 / 服 
务 嚣 数据库 应 用 。 在 Java 1.1 中 ， 这 一 保证 通过 Java 数 据 库 连 接 
(JDBC) 实现 了 。 


数据 库 最 主要 的 一 个 问题 就 是 各 家 公司 之 间 的 规格 大 战 。 确 实 存 在 一 
种 “标准 ”数据 库 语 言 ， 即 “结构 查询 语言 ” (SQL-92) ， 但 通常 都 必须 确 
切 知道 自己 要 和 哪 家 数据 库 公 司 打交道 ， 否 则 极 易 出 问题 ， 尽 管 存在 所 
谓 的 “标准 ”>。JDBC 是 面 癌 “与 平台 无 关 ” 设 计 的 ， 所 以 在 编程 的 时 候 不 
必 关 心 自己 要 使 用 的 是 什么 数据 库 产 品 。 然 而 ， 从 JDBC 里 仍 有 可 能 发 
出 对 某 些 数据 库 公司 专用 功能 的 调用 ， 所 以 仍然 不 可 任性 妄 为 。 


和 Java 中 的 许多 API 一 样 ，JDBC 也 做 到 了 尽量 的 简化 。 我 们 发 出 的 方法 
调用 对 应 于 从 数据 库 收集 数据 时 想当然 的 做 法 : 同 数据 库 连 接 ， 创 建 一 
个 语句 并 执行 得 询 ， 然 后 处 理 结果 集 。 


为 实现 这 一 “与 平台 无 关 ” 的 特点 ，JDBC 为 我 们 提供 了 一 个 “驱动 程序 管 
理 器 *”， 它 能 动态 维护 数据 库 查 询 所 需 的 所 有 驱动 程序 对 象 。 所 以 假如 
要 连接 由 三 家 公司 开发 的 不 同 种 类 的 数据 库 ， 就 需要 三 个 单独 的 驱动 程 
序 对 象 。 驱 动 程序 对 象 会 在 装载 时 由 “驱动 程序 管理 器 ”自动 注册 ， 并 可 
用 Class.forName() 强 行 装载 。 


开 一 个 数据 库 ， 必 须 创 建 一 个 “数据 库 URL”， 它 要 指定 下 述 三 方面 
NA: 
































(1) 用 “jdbc” 指 出 要 使 用 JDBC。 


(2) “THN”: 驱动 程序 的 名 字 或 者 一 种 数据 库 连接 机 制 的 名 称 。 由 于 
JDBC 的 设计 从 ODBC 吸 收 了 许多 灵感 ， 所 以 可 以 选用 的 第 一 种 子 协议 
就 是 “jdbc-odbc 桥 ”， 它 用 “odbc” 关 键 字 即 可 指定 。 


(3) ”数据 库 标识 符 ， 随 使 用 的 数据 库 驱 动 程序 的 不 同 而 变化 ， 但 一 般 都 
提供 了 一 个 比较 符合 逻辑 的 名 称 ， 由 数据 库 管 理 软件 映射 《对 应 ) 到 保 
存 了 数据 表 的 一 个 物理 目录 。 为 使 目 己 的 数据 库 标 识 符 具 有 任何 合 义 ， 

必须 用 自己 的 数据 库 管理 软件 为 自己 喜欢 的 名 字 注 册 【〈 注 册 的 具体 过 程 

















又 随 运 行 平 台 的 不 同 而 变化 ) 。 

所 有 这 些 信 息 都 统一 编译 到 一 个 字 串 里 ， 即 “数据 库 URL”。 举 个 例子 来 
说 ， 和 在 想 通过 ODBC 子 协议 同一 个 标识 为 "people" 的 数据 库 连 接 ， 相 应 
的 数据 库 URL 可 设 为 : 

String dbUrl = "jdbc:odbc:people" 


如 果 通 过 一 个 网 络 连接 ， 数 据 库 URL 也 需要 包含 对 远程 机 器 进行 标识 的 
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Ho 








准备 好 同 数据 库 连 接 后 ， 可 调用 静态 方法 
DriverManager.getConnection()， 将 数据 库 的 URL 以 及 进入 那个 数据 库 所 
需 的 用 户 名 密码 传递 给 它 。 得 到 的 返回 结果 是 一 个 Connection 对 象 ， 利 
用 它 即 可 查询 和 操纵 数据 库 。 


下 面 这 个 例子 将 打开 一 个 联络 信息 数据 库 ， 并 根据 命令 行 提供 的 参数 查 
询 一 个 人 的 姓 (Last Name) 。 它 只 选择 那些 有 E-mail 地 址 的 人 的 名 罕 ， 
然后 列 印 出 符合 查询 条 件 的 所 有 人 : 


//: Lookup.java 


// Looks up email addresses in a 
// local database using JDBC 
import java.sql.*; 
public class Lookup { 
public static void main(String[] args) { 
String dbUrl = "jdbc:odbc:people"; 
String user = ""; 
String password = "": 
try { 


// Load the driver (registers itself) 


Class.forName( 
"sun.jdbc.odbc.JdbcOdbcDriver"); 

Connection c = DriverManager .getConnection( 
dbUrl, user, password); 

Statement s = c.createStatement(); 

// SQL code: 


ResultSet r 


s.executeQuery( 
"SELECT FIRST, LAST, EMAIL " + 
"FROM people.csv people " + 
"WHERE " + 
"(LAST='" + args[0] + "') "+ 
" AND (EMAIL Is Not Null) " + 
"ORDER BY FIRST"); 
while(r.next()) { 
// Capitalization doesn't matter: 
System.out.printin( 
r.getString("Last") + ", " 
+ r.getString("fIRST" ) 
+": " + r.getString("EMAIL") ); 
} 
s.close(); // Also closes ResultSet 


} catch(Exception e) { 


e.printStackTrace(); 


} 
A f/f 


可 以 看 到 ， 数 据 库 URL 的 创建 过 程 与 我 们 前 面 讲 述 的 完全 一 样 。 在 该 例 
中 ， 数 据 库 未 设 密码 保护 ， 所 以 用 户 名 和 和 密码 都 是 空 串 。 


用 DriverManager.getConnection() 建 好 连接 后 ， 接 下 来 可 根据 结果 

Connection 对 象 创建 一 个 Statement (语句 ) 对 象 ， 这 是 用 

createStatement() 方 法 实现 的 。 根 据 结 果 Statement， 我 们 可 调用 

executeQuery()， 问 其 传递 包含 了 SQL-92 标 准 SQL 语 句 的 一 个 字 串 (不 

eae A aye Bx a A), ATA BATE xk FB ARE SQL 
YZ 四) o 


executeQuery() 方 法 会 返回 一 个 ResultSet (HERE) 对 象 ， 它 与 继承 器 非 
TAA: next() 方 法 将 继承 器 移 至 语句 中 的 下 一 条 记录 ; 如 果 已 抵达 结 
果 集 的 末尾 ， 则 返回 null。 我 们 肯定 能 从 executeQuery0 返 回 一 个 
ResultSet 对 象 ， 即 使 查询 结果 是 个 空 集 (也 就 是 说 ， 不 会 产生 一 个 违 
例 ) 。 注 意 在 试图 读 取 任何 记录 数据 之 前 ， 都 必须 调用 一 次 nextO0。 知 
结果 集 为 空 ， 那 么 对 next0 的 这 个 首次 调用 了 束 会 返回 false。 对 于 结果 集 
中 的 每 条 记录 ， 都 可 将 字段 名 作为 字 串 使 用 (当然 还 有 其 他 方法 ) ， 从 
而 选择 不 同 的 字段 。 男 外 要 注意 的 是 字段 名 的 大 小 写 是 无 关 紧 要 的 一 一 
SQL 数据 库 不 在 乎 这 个 问题 。 为 决定 返回 的 类 型 ， 可 调用 getString()， 
getFloat() 等 等 。 到 这 个 时 候 ， 我 们 已 经 用 Java 的 原始 格式 得 到 了 自己 的 
数据 库 数据 ， 接 下 去 可 用 Java 代 码 做 自己 想 做 的 任何 事情 了 。 


15.7.1 让 示例 运行 起 来 

就 JDBC 来 说 ， 代 码 本 身 是 很 容易 理解 的 。 最 令 人 迷惑 的 部 分 是 如 何 使 
它 在 自己 特定 的 系统 上 运行 起 来 。 之 所 以 会 感到 迷惑 ， 是 由 于 它 要 求 我 
们 掌握 如 何 才 能 使 JDBC 驱 动 程序 正确 装载 ， 以 及 如 何 用 我 们 的 数据 库 
管理 软件 来 设置 一 个 数据 库 。 


当然 ， 具 体 的 操作 过 程 在 不 同 的 机 器 上 也 会 有 所 区 别 。 但 这 儿 提 供 的 在 

















ee 助 大 家 理解 在 其 他 平台 上 的 操 


1. 步骤 1: 寻找 JDBC 驱 动 程序 
上 述 程序 包含 了 下 面 这 条 语句 : 
Class.forName("sun.jdbc.odbc.JdbcOdbcDriver"); 


RWB TS ARM, (EKRA BRE Ro To TER FERS 
JDK 1.1 安 装 版 本 中 ， 根 本 不 存在 叫 作 JdbcOdbcDriver.class 的 一 个 文件 。 
所 以 假如 在 看 了 这 个 例子 后 去 寻找 它 ， 那 么 必然 会 徒劳 而 返 。 男 一 些 人 
提供 的 例子 使 用 的 是 一 个 假名 字 ， 如 “myDriver.ClassName”， 但 人 们 从 
字面 上 得 不 到 任何 帮助 。 事 实 上 ， 上 述 用 于 装载 jdbc-odbc 驱 动 程序 〈 实 
际 是 与 JDK ”1.1 配套 提供 的 唯一 驱动 ) 的 语句 在 联机 文档 的 多 处 地 方 均 
有 出 现 〈( 特 别 是 在 一 个 标记 为 JDBC-ODBC Bridge Driver” 的 页 内 ) 。 
大 上 面 的 装载 语句 不 能 工作 ， 那 么 它 的 名 字 可 能 已 随 着 Java 新 版 本 的 发 
布 而 改变 了 ; 此 时 应 到 联机 文档 里 寻找 新 的 表述 方式 。 


各 装载 语句 出 错 ， 会 在 这 个 时 候 得 到 一 个 违例 。 为 了 检验 驱动 程序 装载 
语句 是 不 是 能 正 币 工作 ， 请 将 该 语句 后 面 下 到 catch 从 句 之 间 的 代码 暂时 
“2 如 果 程 序 运行 时 未 出 现 违例 ， 表 明 驱 动 程序 的 装载 是 正确 























2. 步骤 2: 配置 数据 库 


同样 地 ， 我 们 只 限于 在 32 位 Windows 环 境 中 工作 ; 您 可 能 需要 研究 一 下 
目 己 的 操作 系统 ， 找 出 适合 目 己 平台 的 配置 方法 。 


首先 打开 控制 面板 。 其 中 可 能 有 两 个 图 标 都 含有 “ODBC2” 字 样 ， 必 须 选 
择 那 个 “32 位 ODBC”， 因 为 男 一 个 是 为 了 保持 与 16 位 软件 的 回 后 莱 容 而 
设置 的 ， 和 JDBC 混 用 没有 任何 结果 。 双 击 “32 位 ODBC” 图 标 后 ， 看 到 的 
应 该 是 一 个 卡片 式 对 话 框 ， 上 面 一 排 有 多 个 卡片 标签 ， 其 中 包括 “用 户 
DSN”, “系统 DSN”“ 文 件 DSN” 等 等 。 其 中 ,“DSN” 代 表 “ 数 据 源 名 
PK” (Data Source Name) 。 它 们 都 与 JDBC-ODBC 桥 有 关 ， 但 设置 数据 
库 时 唯一 重要 的 地 方 “ 系 统 DSN”。 尺 管 如 此 ， 由 于 需要 测试 自己 的 配置 
以 及 创建 查询 ， 所 以 也 需要 在 “文件 DSN?” 中 设置 自己 的 数据 库 。 这 样 便 
可 让 Microsoft Query LH (Microsoft Office 配 套 提供 〉 正确 地 找到 数 





据 库 。 注 意 一 些 软件 公司 也 设计 了 目 己 的 查询 工具 。 


最 有 趣 的 数据 库 是 我 们 已 经 使 用 过 的 一 个 。 标 准 ODBC 支 持 多 种 文件 格 
式 ， 其 中 包括 由 不 同 公司 专用 的 一 些 格 式 ， 如 dBASE。 然 而 ， 它 也 包括 
了 简单 的 “逗号 分 陋 ASCIT7 格 式 ， 它 几乎 是 每 种 数据 工具 都 能 够 生成 
的 。 束 目前 的 例子 来 说 ， 我 只 选择 自己 的 “people” 数 据 库 。 这 是 我 多 年 
来 一 直 在 维护 的 一 个 数据 库 ， 中 间 使 用 了 各 种 联络 管理 工具 。 我 把 它 导 
出 成 为 一 个 有 逗号 分 隔 的 ASCII 文 件 〈 一 般 有 个 .csv 扩 展 名 ， 用 Outlook 
Express 导 出 通信 短 时 亦 可 选用 同样 的 文件 格式 ) 。 在 “文件 DSN” 区 域 ， 
我 按 下 “添加 ”按钮 ， 选 择 用 于 控制 速 号 分 隔 ASCII 文 件 的 文本 张 动 程序 
(Microsoft Text Driver) ， 然 后 撤消 对 “使 用 当前 目录 ”的 选择 ， 以 便 导 
出 数据 文件 时 可 以 自行 指定 目录 。 


大 家 会 注意 到 在 进行 这 些 工作 的 时 候 ， 并 没有 实际 指定 一 个 文件 ， 只 是 
一 个 目录 。 那 是 因为 数据 库 通 党 是 由 茶 个 目录 下 的 一 系列 文件 构成 的 

《尽管 也 可 能 采用 其 他 形式 ) 。 每 个 文件 一 般 都 包含 了 单个 “数据 表 ”， 

而 且 SQL 语 句 可 以 产生 从 数据 库 中 多 个 表 摘 取出 来 的 结果 《〈 这 叫 作 “ 联 

合 ”， 或 者 join) 只 包含 了 单 张 表 的 数据 库 《〈 就 象 目 前 这 个 ) ny 

作 * 平 面 文件 数据 库 ?。 对 于 大 多 数 问题 ， 如 采 已 经 超过 了 简单 的 数据 存 
储 与 获取 力所能及 的 范围 ， 那 么 必须 使 用 多 个 数据 表 。 通 过 “联合 "， 从 
而 获得 希望 的 结果 。 我 们 把 这 些 叫 作 “ 关 系 型 "数据 库 。 


3. 步骤 3: 测试 配置 
为 了 对 配置 进行 测试 ， 需 用 一 种 方式 核实 数据 库 是 人 否 可 由 得 询 它 的 一 个 


程序 * 见 到 ?”。 当 然 ， 可 以 简单 地 运行 上 述 的 JDBC 示 范 程 序 ， 并 加 入 下 
述 语 句 : 


























Connection c = DriverManager.getConnection( 

dbUrl, user, password); 

右 掷 出 一 个 违例 ， 表 明 你 的 配置 有 误 。 

然而 ， 此 时 很 有 必要 使 用 一 个 自动 化 的 查询 生成 工具 。 我 使 用 的 是 与 
Microsoft Office 配 套 提 供 的 Microsoft Query， 但 你 完全 可 以 自行 选择 一 


个 。 查 询 工 具 必 须知 道 数 据 库 在 什么 地 方 ， 而 Microsoft Query 要 求 我 进 
入 ODBC Administrator 的 “文件 DSN" 卡 片 ， 并 在 那里 新 谎 一 个 条 目 。 同 











样 指定 文本 驱动 程序 以 及 保存 数据 库 的 目录 。 虽 然 可 将 这 个 条 目 命名 为 
目 己 喜欢 的 任何 东西 ， 但 最 好 还 是 使 用 与 “系统 DSN” 中 相同 的 名 字 。 


做 完 这 些 工 作 后 ， 再 用 查询 工具 创建 一 个 新 查询 时 ， 便 会 及 现 上 自己 的 数 
据 库 可 以 使 用 了 。 


4. 步骤 4: 建立 目 己 的 SQL 碍 询 


我 用 Microsoft Query 创 建 的 查询 不 仪 指 出 目标 数据 库存 在 且 次 序 民 好 ， 
也 会 自动 生成 SQL 人 代码， 以便 将 其 插入 我 上 自己 的 Java 程 序 。 我 希望 这 个 
查询 能 够 检查 记录 中 是 否 存在 与 启动 Java 程 序 时 在 命令 行 键入 的 相同 
的 “ 姓 ”(Last Name) 。 所 以 作为 一 个 起 点 ， 我 搜索 自己 的 姓 “Eckel”。 
Jj A 我 希望 只 显示 出 有 对 应 E-mail 地 址 的 那些 名 字 。 创 建 这 个 查询 的 
步骤 如 下 : 


(1) 启动 一 个 新 查询 ， 并 使 用 查询 同 导 (Query Wizard) 。 选 
择 “people” 数 据 库 (等 价 于 用 适应 的 数据 库 URL 打 开 数 据 库 连接 ) 。 


(2) 选择 数据 库 中 的 “people” 表 。 从 这 张 数据 表 中 ， 选 择 FIRST，LAST 和 
EMAIL 列 。 


(3) 在 “Filter Data”( 过 滤器 数据 库 ) 下 ， 选 择 LAST， 并 选 
“equal” (EF) ， 加 上 参数 Eckel。 点 选 “And” 单 选 钮 。 


(4) 选择 EMAIL， 并 选中 “Is not Null”( 不 为 空 )。 














(5) 在 “Sort By” 下 ， 选 择 FIRST。 

得 询 结 末 会 问 我 们 展示 出 是 售 能 得 到 自己 希望 的 东西 。 

现在 可 以 按 下 SQL 按钮 。 不 需要 我 们 任何 方面 的 介入 ， 正 确 的 SQL 代码 
人 以 便 我 们 粘贴 和 复制 。 对 于 这 个 查询 ， 相 应 的 SQL 代 
码 如 下 : 


SELECT people.FIRST, people.LAST, people.EMAIL 





FROM people.csv people 


WHERE (people.LAST='Eckel') AND 
(people.EMAIL Is Not Null) 


ORDER BY people.FIRST 
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互 式 地 测试 自己 的 查询 ， 并 自动 获得 正确 的 代码 。 事 实 上 ， 杀 手 为 这 些 
事情 编码 是 难以 让 人 接受 的 。 

5. 步骤 5: 在 上 自己 的 查询 中 修改 和 粘贴 

我 们 注意 到 上 述 代 码 与 程序 中 使 用 的 代码 是 有 所 区 别 的 。 那 是 由 于 查询 
工具 对 所 有 名 字 都 进行 了 限定 ， 即 便 涉 及 的 仅 有 一 个 数据 表 《〈 知 真 的 涉 
及 多 个 数据 表 ， 这 种 限定 可 避免 来 自 不 同 表 的 同名 数据 列 发 生 冲 突 ) 。 
由 于 这 个 查询 只 需要 用 到 一 个 数据 表 ， 所 以 可 考虑 从 大 多 数 名 字 中 删 

除 *people” 限 定 符 ， 就 象 下 面 这 样 : 


SELECT FIRST, LAST, EMAIL 








FROM people.csv people 
WHERE (LAST='Eckel') AND 
(EMAIL Is Not Null) 


ORDER BY FIRST 


此 外 ， 我 们 不 希望 " 便 编 码 ” 这 个 程序 ， 从 而 只 能 得 找 一 个 特定 的 名 字 。 
相反 ， 它 应 该 能 碍 找 我 们 在 命令 行动 态 提供 的 一 个 名 字 。 所 以 还 要 进行 
必要 的 修改 ， 并 将 SQL 语句 转换 成 一 个 动态 生成 的 字 串 。 如 下 所 未 : 


"SELECT FIRST, LAST, EMAIL " + 





"FROM people.csv people " + 


"WHERE " + 


"(LAST="" 十 args[0] + ie | W + 
" AND (EMAIL Is Not Null) " + 


"ORDER BY FIRST"); 


SQL 还 有 一 种 方式 可 将 名 字 插 入 一 个 查询 ， 名 为 “ 程 
Fe” (Procedures) ， 它 的 速度 非常 快 。 但 对 于 我 们 的 大 多 数 实验 性 数据 
库 操 作 ， 以 及 一 些 初 级 应 用 ， 用 Java 构 建 查询 字 串 已 经 很 不 错 了 。 


从 这 个 例子 可 以 看 出 ， 利 用 目前 找 得 到 的 工具 一 一 特别 是 查询 构建 工具 
涉及 SQL 及 JDBC 的 数据 库 编 程 是 非常 简单 和 直观 的 。 


15.7.2 查找 程序 的 GUI 版 本 


最 好 的 方法 是 让 碍 找 程序 一 直 保 持 运 行 ， 要 碍 找 什 么 东西 时 只 需 简 单 地 
切换 到 它 ， 并 键入 要 查找 的 名 字 即 可 。 下 面 这 个 程序 将 查找 程序 作为 一 
个 “application/applet” 创 建 ， 且 这 加 了 名 字 目 动 填写 功能 ， 所 以 不 必 键 入 
完整 的 姓 ， 即 可 看 到 数据 : 


//: VLookup. java 

















// GUI version of Lookup.java 

import java.awt.*; 

import java.awt.event.*; 

import java.applet.*; 

import java.sql.*; 

public class VLookup extends Applet { 
String dbUrl = "jdbc:odbc:people"; 
String user = ""; 


String password = ""; 


Statement s; 


TextField searchFor = new TextField(20); 


Label completion 
new Label(" 3 
TextArea results = new TextArea(40, 20); 
public void init() { 
searchFor.addTextListener(new SearchForL()); 
Panel p = new Panel(); 
p.add(new Label("Last name to search for:")); 
p.add(searchFor ) ; 
p.add(completion) ; 
setLayout(new BorderLayout()); 
add(p, BorderLayout.NORTH) ; 
add(results, BorderLayout.CENTER) ; 
try { 
// Load the driver (registers itself) 
Class. forName( 
"sun.jdbc.odbc.JdbcOdbcDriver"); 
Connection c = DriverManager .getConnection( 
dbUrl, user, password); 
s = c.createStatement(); 
} catch(Exception e) { 


results.setText(e.getMessage()); 


} 
class SearchForL implements TextListener { 
public void textValueChanged(TextEvent te) { 
ResultSet r; 
if(searchFor.getText().length() == 0) { 
completion.setText(""); 
results.setText(""); 
return; 
} 
try { 
// Name completion: 
r = s.executeQuery( 
"SELECT LAST FROM people.csv people " + 
"WHERE (LAST Like '" + 
searchFor.getText() + 
"%') ORDER BY LAST"); 
if(r.next()) 
completion.setText( 
r.getString("last")); 
r = s.executeQuery( 
"SELECT FIRST, LAST, EMAIL " + 


"FROM people.csv people " + 


"WHERE (LAST='" + 
completion.getText() + 
"') AND (EMAIL Is Not Null) " + 
"ORDER BY FIRST"); 
} catch(Exception e) { 
results.setText( 
searchFor.getText() + "\n"); 
results.append(e.getMessage()); 
return; 


} 


results.setText(""); 
try { 
while(r.next()) { 
results.append( 
r.getString("Last") + ", " 
+ r.getString("fIRST") + 
"ro" + r.,getString("EMAIL") + "\n"); 
} 
} catch(Exception e) { 


results.setText(e.getMessage()); 


public static void main(String[] args) { 
VLookup applet = new VLookup(); 
Frame aFrame = new Frame("Email lookup"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
aFrame.add(applet, BorderLayout .CENTER); 
aFrame.setSize(500, 200); 
applet.init(); 
applet.start(); 
aFrame.setVisible(true); 
} 
} ///:~ 


数据 库 的 许多 逻辑 都 是 相同 的 ， 但 大 家 可 看 到 这 里 添加 了 一 个 
TextListener， 用 于 监视 在 TextField (文本 字段 〉 的 输入。 所 以 只 要 键入 
一 个 新 字符 ， 它 首 先 束 会 试 着 查找 数据 库 中 的 “ 姓 *"， 并 显示 出 与 当前 输 
入 相符 的 第 一 条 记录 (将 其 置 入 completion Label， 并 用 它 作 为 要 查找 的 
MA) 。 因 此 ， 只 要 我 们 键入 了 足够 的 字符 ， 使 程序 能 找到 与 之 相符 的 
Me ——AR ESE, 就 可 以 停 手 了 . 


15.7.3 JDBC AP 为 何如 何 复杂 
阅览 JDBC 的 联机 帮助 文档 时 ， 我 们 往往 会 产生 苦难 情绪 。 特 别 是 


DatabaseMetaData 接 口 一 一 与 Java 中 看 到 的 大 多 数 接口 相反 ， 它 的 体积 
显得 非常 庞大 一 一 存在 着 数量 众多 的 方法 ， 比 如 
dataDefinitionCausesTransactionCommit(), 

getMaxColumnNameLength(), getMaxStatementLength(), 
storesMixedCaseQuotedIdentifiers(), supportsANSI92IntermediateSQL(), 
supportsLimitedOuterJoins() 等 等 。 它 们 有 这 儿 有 什么 意义 吗 ? 


正如 早先 指出 的 那样 ， 数 据 库 起 初 一 直人 处 于 一 种 混乱 状态 。 这 主要 是 由 
于 各 种 数据 库 应 用 提出 的 要 求 造 成 的 ， 所 以 数据 库 工 具 显得 非常 “ 强 

大 "一 一 换言之 , “庞大 ”。 只 是 近 几 年 才 涌 现 出 了 SQL 的 通用 语言 《第 

用 的 还 有 其 他 许多 数据 库 语 言 ) 。 但 即便 象 SQL 这 样 的 “标准 ”>， 也 存在 
无 数 的 变种 ， 所 以 JDBC 必 须 提 供 一 个 巨大 的 DatabaseMetaData 接 口 ， 使 
我 们 的 代码 能 真正 利用 当前 要 连接 的 一 种 “标准 ”SQL 数 据 库 的 能 力 。 简 
言 之 ， 我 们 可 编写 出 简单 的 、 能 移植 的 SQL。 但 如 果 想 优化 代码 的 执行 
那么 为 了 适应 不 同 数据 库 类 型 的 特点 ， 我 们 的 编写 代码 的 奈 烦 就 




















当然 ， 这 并 不 是 Java 的 缺陷 。 数 据 库 产品 之 间 的 差异 是 我 们 和 JDBC 都 
要 面 对 的 一 个 现实 。 但 是 ， 如 果 能 编写 通用 的 查询， 而 不 必 太 关心 性 
能 ， 那 么 事情 就 要 简单 得 多 。 即 使 必须 对 性 能 作 一 番 调 整 ， 只 要 知道 最 
终 面 癌 的 平台 ， 也 不 必 针 对 每 一 种 情况 都 编写 不 同 的 优化 代码 。 


在 Sun 发 布 的 Java 1.1 产 品 中 ， 配 套 提供 了 一 系列 电子 文档 ， 其 中 有 对 
JDBC 更 全 面 的 介绍 。 此 外 ， 在 由 Hamilton Cattel 和 Fisher 编 兰 、Addison- 
Wesley 于 1997 年 出 版 的 《JDBC Database Access with Java》 中 ， 也 提供 
了 有 关 这 一 主题 的 许多 有 用 资料 。 同 时 ， 书 店 里 也 经 常 出 现 一 些 有 关 
JDBC 的 新 书 。 


15.8 远程 方法 


为 通过 网 络 执行 其 他 机 器 上 的 代码 ， 传 统 的 方法 不 仅 难以 学 习 和 和 掌握 ， 
也 极 易 出 错 。 思 考 这 个 问题 最 佳 的 方式 是 : 作 些 对 象 正 好 位 于 为 一 台 机 
器 ， 我 们 可 加 它们 发 送 一 条 消息 ， 并 获得 返回 结果 ， 就 象 那些 对 象 位 于 
目 己 的 本 地 机 器 一 样 。Java 1.1 的 “远程 方法 调用 ”(RMI) 采用 的 正 是 这 
种 抽象 。 本 节 将 引导 大 家 经 历 一 些 必要 的 步骤 ， 创 建 目 己 的 RMI 对 象 。 


15.8.1 远程 接口 概念 

RMI 对 接口 有 着 强烈 的 依赖 。 在 需要 创建 一 个 远程 对 象 的 时 候 ， 我 们 通 
过 传递 一 个 接口 来 隐藏 基层 的 实施 细节 。 所 以 客户 得 到 远程 对 象 的 一 个 
句柄 时 ， 它 们 真正 得 到 的 是 接口 句柄 。 这 个 句柄 正好 同一 些 本 地 的 根 代 
码 连 接 ， 由 后 者 负责 通过 网 络 通信 。 但 我 们 并 不 关心 这 些 事情 ， 只 需 通 
过 自己 的 接口 句柄 发 送 消息 即 可 。 

创建 一 个 远程 接口 时 ， 必 须 遵 守 下 列 规则 : 

(1) ”远程 接口 必须 为 public 属 性 (不 能 有 “ 包 访 问 *”， 也 就 是 说 ， 它 不 能 
是 “友好 的 ”) 。 否 则 ， 一 旦 客户 试图 北 载 一 个 实现 了 远程 接口 的 远程 对 
象 ， 就 会 得 到 一 个 错误 。 

(2) 远程 接口 必须 扩展 接口 java.rmi.Remote。 


(3)” 除 与 应 用 程序 本 和 映 有 关 的 违例 之 外 ， 远 程 接口 中 的 每 个 方法 都 必须 
在 自己 的 throws 从 人 句 中 声明 java.rmi.RemoteException。 


(4) “作为 参数 或 返回 值 传递 的 一 个 远程 对 象 〈 不 管 是 直接 的 ， 还 是 在 本 
HEM RPA) 必须 声明 为 远程 接口 ， 不 可 声明 为 实施 类 。 


下 面 是 一 个 简单 的 远程 接口 示例 ， 它 代表 的 是 一 个 精确 计时 服务 : 


//: PerfectTimeI,java 




















// The PerfectTime remote interface 


package c15.ptime; 


import java.rmi.*; 
interface PerfectTimeI extends Remote { 
long getPerfectTime() throws RemoteException; 
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它 表 面 上 与 其 他 接口 是 类 似 的 ， 只 是 对 Remote 进 行 了 扩展 ， 而 且 它 的 所 
有 方法 都 会 “ 接 ” 出 RemoteException 〈 远 程 违 例 ) 。 记 住 接 口 和 它 所 有 的 
方法 都 是 public 的 。 








15.8.2 远程 接口 的 实施 

服务 器 必须 包含 一 个 扩展 了 UnicastRemoteObject 的 类 ， 并 实现 远程 接 
口 。 这 个 类 也 可 以 含有 附加 的 方法 ， 但 客户 只 能 使 用 远程 接口 中 的 方 
法 。 这 是 显然 的 ， 因 为 客户 得 到 的 只 是 指 问 接 口 的 一 个 句柄 ， 而 非 实现 


它 的 那个 类 。 


必须 为 远程 对 象 明 确定 义 构 建 器 ， 即 使 只 准备 定义 一 个 默认 构建 器 ， 用 
它 调 用 基础 类 构建 问 。 必 须 把 它 明确 地 编写 出 来 ， 因 为 它 必 须 “ 掷 ?出 
RemoteException 违 例 。 


下 面 列 出 远程 接口 PerfectTime 的 实施 过 程 : 


//: PerfectTime.java 


// The implementation of the PerfectTime 
// remote object 

package c15.ptime; 

import java.rmi.*; 

import java.rmi.server.*; 

import java.rmi.registry.*; 


import java.net.*; 


public class PerfectTime 
extends UnicastRemoteObject 
implements PerfectTimel { 
// Implementation of the interface: 
public long getPerfectTime() 
throws RemoteException { 
return System.currentTimeMillis(); 
} 
// Must implement constructor to throw 
// RemoteException: 
public PerfectTime() throws RemoteException { 
// super(); // Called automatically 
} 
// Registration for RMI serving: 
public static void main(String[] args) { 
System.setSecurityManager( 
new RMISecurityManager()); 
try { 
PerfectTime pt = new PerfectTime()/; 
Naming. bind( 
"//colossus:2005/PerfectTime", pt); 
System.out.println("Ready to do time"); 


} catch(Exception e) { 


e.printStackTrace(); 


} 
A f/f 


在 这 里 ，main0 控 制 着 设置 服务 器 的 全 部 细节 。 保 存 RMI 对 象 时 ， 必 须 
在 程序 的 菏 个 地 方 采 取 下 述 操作 : 


(1) 创建 和 安装 一 个 安全 管理 器 ， 令 其 支持 RMI。 作 为 Java 发 行 包 的 一 部 
分 ， 适 用 于 RMI 唯 一 一 个 是 RMISecurityManager。 


(2) 创建 远程 对 象 的 一 个 或 多 个 实例 。 在 这 里 ， 大 家 可 看 到 创建 的 是 
PerfectTime 对 象 。 


(3)” 问 RMI 远 程 对 象 注册 表 注 册 至 少 一 个 远程 对 象 。 一 个 远程 对 象 拥有 
的 方法 可 生成 指向 其 他 远程 对 象 的 句柄 。 这 样 一 来 ， 客 户 只 需 到 注册 表 
里 访问 一 次 ， 得 到 第 一 个 远程 对 象 即 可 。 


1. 设置 注册 表 
在 这 儿 ， 大 家 可 看 到 对 静态 方法 Naming.bind() 的 一 个 调用 。 然 而 ， 这 个 


调用 要 求 注册 表 作 为 计算 机 上 的 一 个 独立 进程 运行 。 注 册 表 服务 器 的 名 
字 是 rmiregistry。 在 32 位 Windows 环 境 中 ， 可 使 用 : 











start rmiregistry 
令 其 在 后 台 运 行 。 在 Unix 中 ， 使 用 : 
rmiregistry & 


和 许多 网 络 程序 一 样 ，rmiregistry 位 于 机 器 启动 它 所 在 的 某 个 IP 地 址 
处 ， 但 它 也 必须 监视 一 个 端口 。 如 果 象 上 面 那样 调用 rmiregistry， 不 使 
用 参数 ， 注 册 表 的 端口 就 会 默认 为 1099。 知 希望 它 位 于 其 他 某 个 端口 ， 
只 需 在 命令 行 添加 一 个 参数 ， 指 定 那 个 端口 编号 即 可 。 对 这 个 例子 来 
说 ， 端 口 将 位 于 2005， 所 以 rmiregistry 应 该 象 下 面 这 样 启 动 (对 于 32 位 
Windows) : 





start rmiregistry 2005 
对 于 Unix， 则 使 用 下 述 命令 : 
rmiregistry 2005 & 


与 端口 有 关 的 信息 必须 传送 给 bind() 命 令 ， 同 时 传送 的 还 有 注册 表 所 在 
的 那 台 机 喜 的 耳 地 址 。 但 假名 我 们 想 在 本 地 测试 RMI 程 序 ， 就 象 本 章 的 
网 络 程序 一 直 测 试 的 那样 ， 这 样 做 就 会 之 来 问题 。 在 JDK 1.1.1 版 本 中 ， 
存在 着 下 述 两 方面 的 问题 (注释 中 ): 


(1) localhost 不 能 随 RMI 工 作 。 所 以 为 了 在 单独 一 台 机 器 上 完成 对 RMI 的 
测试 ， 必 须 提供 机 器 的 名 字 。 为 了 在 32 位 Windows 坏 境 中 调查 自己 机 器 
的 名 字 ， 可 进入 控制 面板 ， 选 择 “ 网 络 ”， 选 择 “ 标 识 ” 卡 片 ， 其 中 列 出 了 
计算 机 的 名 字 。 就 我 自己 的 情况 来 说 ， 我 的 机 器 叫 作 “Colossus”( 因 为 
我 用 几 个 大 容量 的 硬盘 保存 各 种 不 同 的 开发 系统 一 一 Clossus 是 “巨人 ”的 
意思 ) 。 似 乎 大 写 形式 会 被 忽略 。 


(2) ”除非 计算 机 有 一 个 活动 的 TCP/IP 连 接 ， 否 则 RMI 不 能 工作 ， 即 使 所 
有 组 件 都 只 需要 在 本 地 机 器 里 互相 通信 。 这 意味 着 在 试图 运行 程序 之 
前 ， 必 须 连 接 到 目 己 的 ISP《〈 因 特 网 服务 提供 者 ) ， 人 否则 会 得 到 一 些 含 
义 模 糊 的 违例 消息 。 

O: 为 找 出 这 些 信息 ， 我 不 知 损伤 了 多 少 个 脑 细胞 。 

考虑 到 这 些 因 系 ，bind0 命 令 变 成 了 下 面 这 个 样子 : 
Naming.bind("//colossus:2005/PerfectTime", pt); 
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Naming.bind("//colossus/PerfectTime", pt); 


在 JDK 未 来 的 版 本 中 “(1.1 之后)》， 一 旦 改正 了 localhost 的 问题 ， 就 能 正 
常 地 进行 本 地 测试 ， 去 掉 人 P 地 址 ， 只 使 用 标识 符 : 


Naming.bind("PerfectTime", pt); 
服务 名 是 任意 的 ， 它 在 这 里 正好 为 PerfectTime， 和 类 名 一 样 ， 但 你 可 以 











根据 情况 任意 修改 。 最 重要 的 是 确保 它 在 注册 表 里 是 个 独一无二 的 名 

字 ， 以 便 客 户 正 常 地 获取 远程 对 象 。 若 这 个 名 字 已 在 注册 表 里 了 ， 就 会 
得 到 一 个 AlreadyBoundException 违 例 。 为 防止 这 个 问题 ， 可 考虑 坚持 使 
用 rebind0， 放 弃 bind0)。 这 是 由 于 rebindO0 要 么 会 添加 一 个 新 条 目 ， 要 么 
将 同名 的 条 目 蔡 换 掉 。 


尽管 main0 退 出 ， 我 们 的 对 象 已 经 创建 并 注册 ， 所 以 会 由 注册 表 一 直 保 
持 活 动 状 态 ， 等 候 客户 到 达 并 发 出 对 它 的 请 求 。 只 要 rmiregistry 处 于 运 

行 状态 ， 而 且 我 们 没有 为 名 字 调 用 Naming.unbind() 方 法 ， 对 象 就 肯定 位 
于 那个 地 方 。 考 虑 到 这 个 原因 ， 在 我 们 设计 自己 的 代码 时 ， 需 要 先 关 闭 
rmiregistry， 并 在 编译 远程 对 象 的 一 个 新 版 本 时 重新 启动 它 。 


并 不 一 定 要 将 rmiregistry 作 为 一 个 外 部 进程 局 动 。 知 事前 知道 自己 的 是 
要 求 用 以 注册 表 的 唯一 一 个 应 用 ， 束 可 在 程序 内 部 局 动 它 ， 使 用 下 述 代 
AS: 




















LocateRegistry.createRegistry(2005); 


和 前 面 一 样 ，2005 代 表 我 们 在 这 个 例子 里 选用 的 端口 号 。 这 等 价 于 在 命 
令 行 执行 rmiregistry 2005。 但 在 设计 RMI 代 码 时 ， 这 种 做 法 往往 显得 更 
加 方便 ， 因 为 它 取消 了 局 动 和 中 止 注册 表 所 需 的 额外 步骤 。 一 旦 执行 完 
这 个 代码 ， 就 可 象 以 前 一 样 使 用 Naming 进 行 “ 绑 定 ”一 一 bind()。 


15.8.3 创建 根 与 干 


知 编译 和 运行 PerfectTime.java， 即 使 rmiregistry 正 确 运行 ， 它 也 无 法 工 
作 。 这 是 由 于 RMI 的 框架 尚未 就 位 。 首 先 必 须 创 建 根 和 和 王 ， 以 便 提 供 网 
络 连 接 操作 ， 并 使 我 们 将 远程 对 象 伪装 成 自己 机 器 内 的 某 个 本 地 对 象 。 


所 有 这 些 幕 后 的 工作 都 是 相当 复杂 的 。 我 们 从 远程 对 象 传 入 、 传 出 的 任 
何 对 象 都 必须 “implement Serializable”( 如 果 想 传递 远程 引用 ， 而 非 整个 
对 象 ， 对 象 的 参数 就 可 以 “implement Remote”) 。 因 此 可 以 想象 ， 当 根 
和 干 通 过 网 络 “ 汇 集 ” 所 有 参数 并 返回 结果 的 时 候 ， 会 自动 进行 序列 化 以 
及 数据 的 重新 装配 。 笠 运 的 是 ， 我 们 根本 没 必 要 了 解 这 些 方面 的 任何 细 
祈 ， 但 根 和 干 却 是 必须 创建 的 。 一 个 简单 的 过 程 如 下 : 在 编译 好 的 代码 
中 调用 rmic， 它 会 创建 必需 的 一 些 文件 。 所 以 唯一 要 做 的 事情 就 是 为 编 
译 过 程 新 添 一 个 步骤 。 




















然而 ，rmic 工 具 与 特定 的 包 和 类 路 径 有 很 大 的 关联 。PerfectTime.java 位 
于 包 c15.Ptime 中 ， 即 使 我 们 调用 与 PerfectTime.class 同 一 目录 内 的 rmic， 
rmic 都 无 法 找到 文件 。 这 是 由 于 它 搜 索 的 是 类 路 径 。 因 此 ， 我 们 必须 同 
IN FREER IE, WER RIK: 


rmic c15.PTime.PerfectTime 


执行 这 个 命令 时 ， 并 不 一 定 非 要 在 包含 了 PerfectTime.class 的 目录 中 ， 但 
结果 会 置 于 当前 目录 。 


若 rmic 成 功 运 行 ， 日 录 里 就 会 多 出 两 个 新 类 : 








PerfectTime_Stub.class 


PerfectTime_Skel.class 


它们 分 别 对 应 根 (Stub) MF (Skeleton) 。 现 在 ， 我 们 已 准备 好 让 服 
务 器 与 客户 互相 沟通 了 。 


15.8.4 使 用 远程 对 象 


RMI 全 部 的 守则 就 是 尽 可 能 简化 远程 对 象 的 使 用 。 我 们 在 客户 程序 中 要 
做 的 唯一 一 件 额 外 的 事情 就 是 查找 并 从 服务 器 取 回 远程 接口 。 自 此 以 
后 ， 剩 下 的 事情 就 是 普通 的 Java 编 程 : 将 消息 发 给 对 象 。 下 面 是 使 用 
PerfectTime 的 程序 : 














//: DisplayPerfectTime. java 


// Uses remote object PerfectTime 
package ci5.ptime; 

import java.rmi.*; 

import java.rmi.registry.*; 
public class DisplayPerfectTime { 


public static void main(String[] args) { 


System.setSecurityManager( 
new RMISecurityManager()); 
try { 
PerfectTimeI t = 
(PerfectTimel )Naming. lookup( 
"//colossus:2005/PerfectTime" ); 
for(int i = 0; i < 10; i++) 
System.out.printin("Perfect time = " + 
t.getPerfectTime()); 
} catch(Exception e) { 


e.printStackTrace(); 


$ 


} ///:~ 


ID 字 串 与 那个 用 Naming 注 册 对 象 的 那个 字 串 是 相同 的 ， 第 一 部 分 指出 
人 
上 的 一 台 机 器 。 


从 Naming.lookupO 返 回 的 必须 造型 到 远程 接口 ， 而 不 是 到 类 。 知 换 用 
类 ， 会 得 到 一 个 违例 提示 。 


在 下 述 方法 调用 中 : 
t.getPerfectTime( ) 


我 们 可 看 到 一 旦 获得 远程 对 象 的 句柄 ， 用 它 进 行 的 编程 与 用 本 地 对 象 的 
编程 是 非常 相似 〈 仅 有 一 个 区 别 : 远程 方法 会 < 掷 ? 出 一 个 





RemoteException 违 例 ) 。 
15.8.5 RMI 的 蔡 选 方案 


RMI 只 是 一 种 创建 特殊 对 象 的 方式 ， 它 创建 的 对 象 可 通过 网 络 发 布 。 它 
最 大 的 优点 就 是 提供 了 一 种 “ 纯 Java” 方 案 ， 但 假如 已 经 有 许多 用 其 他 语 
言 编 写 的 代码 ， 则 RMI 可 能 无 法 满足 我 们 的 要 求 。 目 前 ， 两 种 最 具 苋 争 
力 的 普选 方案 是 微软 的 DCOM (根据 微软 的 计划 ， 它 最 终 会 移植 到 除 
Windows 以 外 的 其 他 平台 ) 以 及 CORBA。CORBA 自 Java ”1.1 便 开始 支 
持 ， 是 一 种 全 新 设计 的 概念 ， 面 向 器 平台 应 用 。 在 由 Orfali 和 Harkey 编 
341) «Client/Server Programming with Java and CORBA》 一 书 中 (John 
Wiley&Sons 1997 年 出 版 ) ， 大 家 可 获得 对 Java 中 的 分 布 式 对 象 的 全 面 介 
绍 (该 书 似乎 对 CORBA 似 乎 有 些 偏见 ) 。 为 CORBA 赋 予 一 个 较 公 正 的 
对 待 的 一 本 书 是 由 Andreas Vogel 和 Keith Duddy 编 号 的 《Java 
Programming with CORBA) , John Wiley&Sons 于 1997 年 出 版 。 





15.9 总 结 


由 于 篇 幅 所 限 ， 还 有 其 他 许多 涉及 连 网 的 概念 没有 介绍 给 大 家 。Java 也 
为 URL 提 供 了 相当 全 面 的 文 持 ， 包 括 为 因特网 上 不 同类 型 的 客户 提供 协 
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除 此 以 外 ， 一 种 正在 逐步 流行 的 技术 叫 作 Servlet Server。 它 是 一 种 因 特 
网 服务 器 应 用 ， 通 过 Java 控 制 客户 请 求 ， 而 非 使 用 以 前 那 种 速度 很 慢 、 
有 旦 相当 底 烦 的 CGI (通用 网 关 接 口 ) 协议 。 这 意味 着 为 了 在 服务 器 那 一 
端 提供 服务 ， 我 们 可 以 用 Java 编 程 ， 不 必 使 用 自己 不 熟悉 的 其 他 语言 。 
pe 的 移植 能 力 ， 所 以 不 必 关 心 具 体 容纳 这 个 服务 器 是 什 
么 交合 。 


所 有 这 些 以 及 其 他 特性 都 在 《Java Network Programming》 一 书 中 得 到 
了 详细 讲述 。 该 书 由 Elliotte Rusty Harold 编 著 ，O'Reilly 于 1997 年 出 版 。 








15.10 练习 


(1) 编译 和 运行 本 章 中 的 JabberServer 和 JabberClient 程 序 。 接 着 编辑 一 下 
程序 ， 删 去 为 输入 和 输出 设计 的 所 有 缓冲 机 制 ， 然 后 再 次 编译 和 运行 ， 
WEE— FR. 


(2) 创建 一 个 服务 器 ， 用 它 请 求 用 户 输入 密码 ， 然 后 打开 一 个 文件 ， 并 
将 文件 通过 网 络 连 接 传送 出 去 。 创 建 一 个 同 该 服务 器 连接 的 客户 ， 为 其 
分 配 适 当 的 密码 ， 然 后 捕获 和 保存 文件 。 在 自己 的 机 器 上 用 

localhost (通过 调用 InetAddress.getByName(null) 生 成 本 地 IP 地 址 
127.0.0.1) 测试 这 两 个 程序 。 


(3) 修改 练习 2 中 的 程序 ， 令 其 用 多 线程 机 制 对 多 个 客户 进行 控制 。 
(4) 修改 JabberClient， 禁 止 输出 刷新 ， 并 观察 结果 。 


(5) ”以 ShowHTML.java 为 基础 ， 创 建 一 个 程序 片 ， 令 其 成 为 对 自己 Web 
站 点 的 特定 部 分 进行 密码 保护 的 大 门 。 


(6) 《可 能 有 些 难度 ) 创建 一 对 客户 / 服务 器 程序 ， 利 用 数据 报 
(Datagram) 将 一 个 文件 从 一 合 机 器 传 到 另 一 侣 《参见 本 章 数 据 报 小 节 
末尾 的 叙述 ) 。 


(7) (可 能 有 些 难 度 ) 对 VLookup.java 程 序 作 一 番 修 改 ， 使 我 们 能 点 击 得 
到 的 结果 名 字 ， 然 后 程序 会 自动 取得 那个 名 字 ， 并 把 它 复制 到 剪贴 板 

(以 便 我 们 方便 地 粘贴 到 自己 的 E-mail) 。 可 能 要 回 过 头 去 研究 一 下 IO 
数据 流 的 那 一 章 ， 回 忆 访 如何 使 用 Java 1.1 te. 


























16H 设计 范式 


本 章 要 问 大 家 介绍 重要 但 却 并 不 是 那么 传统 的 “范式 ”(Pattern) 程序 设 
计 方 法 。 


在 向 面 癌 对 象 程序 设计 的 演化 过 程 中 ， 或 许 最 重要 的 一 步 就 是 “设计 范 

式 ”(Design Pattern) 的 问世 。 它 在 由 Gamma，Helm 和 Johnson 编 车 的 
《Design Patterns》 一 书 中 被 定义 成 一 个 “里 程 碑 ”( 该 书 由 Addison- 
Wesley 于 1995 年 出 版 ， 注 释 四 ) 。 那 本 书 列 出 了 解决 这 个 问题 的 23 种 不 
同 的 方法 。 在 本 章 中 ， 我 们 准备 伴随 几 个 例子 揭示 出 设计 范式 的 基本 概 
念 。 这 或 许 能 激 起 您 阅读 《Design Patten) BRAE. FKE, ME 
书 现在 已 成 为 几乎 所 有 OOP 程 序 员 都 必 备 的 参考 书 。 


©: 但 警告 大 家 : 书 中 的 例子 是 用 C++ 写 的 。 


本 章 的 后 一 部 分 包含 了 展示 设计 进化 过 程 的 一 个 例子 ， 首 先是 比较 原始 
的 方案 ， 经 过 逐渐 发 展 和 改进 ， 慢 慢 成 为 更 符合 馆 辑 、 更 为 恰当 的 设 
计 。 访 程序 〈 仿 真 垃圾 分 类 ) 一 直 都 在 进化 ， 可 将 这 种 进化 作为 目 己 设 
计 方 案 的 一 个 原型 一 一 先 为 特定 的 问题 提出 一 个 适当 的 方案 ， 再 逐步 改 
普 ， 使 其 成 为 解决 那 类 问题 一 种 最 灵活 的 方案 。 














16.1 范式 的 概念 


在 最 开始 ， 可 将 范式 想象 成 一 种 特别 聪明 、 能 够 自我 适应 的 手法 ， 它 可 
以 解决 特定 类 型 的 问题 。 也 就 是 说 ， 它 类 似 一 些 需 要 全 面 认识 茶 个 问题 
的 人 。 在 了 解 了 问题 的 方方面面 以 后 ， 最 后 提出 一 套 最 通用 、 最 灵活 的 
解决 方案 。 具 体 问题 或 许 是 以 前 见 到 并 解决 过 的 。 然 而 ， 从 前 的 方案 也 
许 并 不 是 最 完善 的 ， 大 家 会 看 到 它 如 何在 一 个 范式 里 具体 表达 出 来 。 


尽管 我 们 称 之 为 “设计 范式 "， 但 它们 实际 上 并 不 局 限于 设计 领域 。 思 
考 “ 范 式 * 时 ， 应 脱离 传统 意义 上 分 析 、 设 计 以 及 实施 的 思考 方式 。 相 
反 , “范式 ”是 在 一 个 程序 里 具体 表达 一 套 完 整 的 思想 ， 所 以 它 有 时 可 能 
出 现在 分 析 阶 段 或 者 高 级 设计 阶段 。 这 一 点 是 非常 有 趣 的 ， 因 为 范式 具 
有 以 代码 形式 直接 实现 的 形式 ， 所 以 可 能 不 希望 它 在 低级 设计 或 者 具体 
实施 以 前 显露 出 来 〈 而 且 事实 上 ， 除 非 真正 进入 那些 阶段 ， 否 则 一 般 意 
识 不 到 自己 需要 一 个 范式 来 解决 问题 ) 。 


范式 的 基本 概念 亦 可 看 成 是 程序 设计 的 基本 概念 : 谎 加 一 层 新 的 抽象 ! 

只 要 我 们 抽象 了 某 些 东西 ， 就 相当 于 隔离 了 特定 的 细节 。 而 且 这 后 面 最 

引 人 注 目的 动机 束 是 “将 保持 不 变 的 东西 刁 上 发 生 的 变化 孤立 出 来 "。 这 

样 做 的 另 一 个 原因 是 一 旦 发 现 程序 的 茶 部 分 由 于 这 样 或 那样 的 原因 可 能 

发 生变 化 ， 我 们 一 般 都 想 防止 那些 改变 在 代码 内 部 繁衍 出 其 他 变化 。 这 
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为 设计 出 功能 强大 且 易 于 维护 的 应 用 项 目 ， 通 常 最 困难 的 部 分 就 是 找 出 
我 称 之 为 “领头 变化 ”的 东西 。 这 意味 着 需要 找 出 造成 系统 改变 的 最 重要 
的 东西 ， 或 者 换 一 个 角度 ， 找 出 付出 代价 最 高 、 开 销 最 大 的 那 一 部 分 。 
本 
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所 以 设计 范式 的 最 终 目 标 就 是 将 代码 中 变化 的 内 容 隔离 开 。 如 果 从 这 个 
角度 观察 ， 就 会 发 现 本 书 实际 已 采用 了 一 些 设计 范式 。 举 个 例子 来 说 ， 
继承 可 以 想象 成 一 种 设计 范式 《类似 一 个 由 编译 需 实 现 的) 。 在 都 拥有 
同样 接口 〈 即 保持 不 变 的 东西 ) 的 对 象 内 部 ， 它 允许 我 们 表达 行为 上 的 
差异 〈 即 发 生变 化 的 东西 》。 合 成 亦 可 想象 成 一 种 范式 ， 因 为 它 允 许 我 
们 修改 一 一 动态 或 静态 一 一 用 于 实现 类 的 对 象 ， 所 以 也 能 修改 类 的 运作 
































方式 。 

在 《Design Patterns》 一 书 中 ， 大 家 还 能 看 到 另 一 种 范式 : “继承 器 ”《〈 即 
Iterator, Java ”1.0 和 1.1 不 负责 任 地 把 它 叫 作 Enumeration， 即 “ 枚 举 ”; 
Javal.2 的 集合 则 改 回 了 ”继承 器 ”的 称呼 ) 。 当 我 们 在 集合 里 过 历 ， 逐 个 
选择 不 同 的 元 系 时 ， 继 承 器 可 将 集合 的 实施 细节 有 效 地 隐藏 起 来 。 利 用 
继承 器 ， 可 以 编写 出 通用 的 代码 ， 以 便 对 一 个 序列 里 的 所 有 元 素 采 取 某 
种 操作 ， 同 时 不 必 关 心 这 个 序列 是 如 何 构建 的 。 这 样 一 来 ， 我 们 的 通用 
代码 即 可 伴随 任何 能 产生 继承 占 的 集合 使 用 。 


16.1.1 单子 

或 许 最 简单 的 设计 范式 就 是 “单子 ”(Singleton) ， 它 能 提供 对 象 的 一 个 
(而 且 只 有 一 个 ) 实例。 单子 在 Java 库 中 得 到 了 应 用 ， 但 下 面 这 个 例子 
显得 更 直接 一 些 : 


//: SingletonPattern. java 




















// The Singleton design pattern: you can 
// never instantiate more than one. 
package c16; 
// Since this isn't inherited from a Cloneable 
// base class and cloneability isn't added, 
// making it final prevents cloneability from 
// being added in any derived classes: 
final class Singleton { 
private static Singleton s = new Singleton(47); 
private int 1; 
private Singleton(int x) { i = x; } 


public static Singleton getHandle() { 


return s; 
ae 
public int getValue() { return i; } 
public void setValue(int x) { i= x; } 
} 
public class SingletonPattern { 
public static void main(String[] args) { 
Singleton s = Singleton.getHandle(); 
System.out.println(s.getValue()); 
Singleton s2 = Singleton.getHandle(); 
s2.setValue(9); 
System.out.println(s.getValue()); 
try { 
// Can't do this: compile-time error. 
// Singleton s3 = (Singleton)s2.clone(); 
} catch(Exception e) {} 
} 
} ///:~ 








创建 单子 的 关键 就 是 防止 客户 程序 员 玉 用 除 由 我 们 提供 的 之 外 的 任何 一 
种 方式 来 创建 一 个 对 象 。 必 须 将 所 有 构建 器 都 设 为 private〈( 私 有) ， 而 
且 至 少 要 创建 一 个 构建 器 ， 以 防止 编译 器 帮 我 们 自动 同步 一 个 默认 构建 
器 〔( 它 会 自 做 聪明 地 创建 成 为 “友好 的 ”一 一 friendly， 而 非 private〉。 


此 时 应 决定 如 何 创建 自己 的 对 象 。 在 这 儿 ， 我 们 选择 了 静态 创建 的 方 





式 。 但 亦 可 选择 等 候 客户 程序 员 发 出 一 个 创建 请 求 ， 然 后 根据 他 们 的 要 
求 动态 创建 。 不 管 在 哪 种 情况 下 ， 对 象 都 应 该 保存 为 “私有 ”属性 。 我 们 
通过 公用 方法 提供 访问 途径 。 在 这 里 ，getHandleO 会 产生 指向 Singleton 
的 一 个 句柄 。 剩 下 的 接口 〈getValue0 和 setValue0 ) 属于 普通 的 类 接 
Ho 


Java 也 允许 通过 克隆 (Clone) 方式 来 创建 一 个 对 象 。 在 这 个 例子 中 ， 将 
类 设 为 final 可 禁止 元 隆 的 发 生 。 由 于 Singleton 是 从 Object 直 接 继承 的 ， 

所 以 clone() 方 法 会 保持 protected〈 受 保护 ) 属性 ， 不 能 够 使 用 它 〈 强 行 
使 用 会 造成 编译 期 错误 ) 。 然 而 ， 假 如 我 们 是 从 一 个 类 结构 中 继承 ， 那 
个 结构 已 经 过 载 了 done() 方 法 ， 使 其 具有 public 属 性 ， 并 实现 了 
Cloneable， 那 么 为 了 禁止 克隆， 需要 过 载 clone()， 并 撕 出 一 个 
CloneNotSupportedException 〈 不 文 持 克隆 违例 ) ， 束 象 第 12 章 介绍 的 那 
样 。 亦 可 过 载 cone0， 并 简单 地 返回 this。 那 样 做 会 造成 一 定 的 混 消 ， 
因为 客户 程序 员 可 能 错误 地 认为 对 象 肖 未 殉 隆 ， 仍 然 操 纵 的 是 原来 的 慎 
广 。 


注意 我 们 并 不 限于 只 能 创建 一 个 对 象 。 亦 可 利用 该 技术 创建 一 个 有 限 的 
对 象 池 。 但 在 那 种 情况 下 ， 可 能 需要 解决 池内 对 象 的 共享 问题 。 如 有 果 不 
地 真 的 过 到 这 个 问题 ， 可 以 自己 设计 一 套 方案 ， 实 现 共享 对 象 的 登记 与 
撤消 登记 。 

16.1.2 范式 分 类 

(Design Patterns》 一 书 讨 论 了 23 种 不 同 的 范式 ， 并 依据 三 个 标准 分 类 
《所 有 标准 都 涉及 那些 可 能 发 生变 化 的 方面 ) 。 这 三 个 标准 是 : 


(1) 创建 : 对象 的 创建 方式 。 这 通常 涉及 对 象 创 建 细节 的 隔离 ， 这 样 便 
不 必 依 赖 具 体 类 型 的 对 象 ， 所 以 在 新 添 一 种 对 象 类 型 时 也 不 必 改 动 代 
码 。 

















(2) ”结构 :设计 对 象 ， 满 足 特定 的 项 目 限 制 。 这 涉及 对 象 与 其 他 对 象 的 
连接 方式 ， 以 保证 系统 内 的 改变 不 会 影响 到 这 些 连 接 。 


(3) 行为 : 对 程序 中 特定 类 型 的 行动 进行 操纵 的 对 象 。 这 要 求 我 们 将 希 
望 采 取 的 操作 封装 起 来 ， 比 如 解释 一 种 语言 、 实 现 一 个 请 求 、 在 一 个 序 
列 中 遍历 (就 象 在 继承 器 中 那样 ) 或 者 实现 一 种 算法 。 本 章 提供 了 “ 观 
gees” (Observer) 和 “访问 器 ”(CVisitor〉 的 范式 的 例子 。 





(Design Patterns》 为 所 有 这 23 种 范式 都 分 别 使 用 了 一 节 ， 随 附 的 还 有 
大 量 示 例 ， 但 大 多 是 用 C++ 编写 的 ， 少 数 用 Smalltalk 编 号 〈 如 看 过 这 本 
书 ， 束 知道 这 实际 并 不 是 个 大 问题 ， 因 为 很 容易 即 可 将 基本 概念 从 两 种 
语言 翻译 到 Java 里 ) 。 现 在 这 本 书 并 不 打算 重复 《Design Patterns》 介 绍 
的 所 有 范式 ， 因 为 那 是 一 本 独立 的 书 ， 大 家 应 该 单独 阅读 。 相 反 ， 本 和 章 
只 准备 给 出 一 些 例 子 ， 让 大 家 先 对 范式 有 个 大 致 的 印象 ， 并 理解 它们 的 
重要 性 到 底 在 哪里 。 





16.2 观察 器 范式 


观 罕 器 (Observer) 范式 解雇 的 是 一 个 相当 普通 的 问题 : 由 于 某 些 对 象 
的 状态 发 生 了 改变 ， 所 以 一 组 对 象 都 需要 更 新 ， 那 么 该 如 何 解 决 ? 在 
Smalltalk 的 MVC 模型 一 视图 一 控制 器 〉 的 “模型 一 视图 ”部 分 中 ， 或 在 
几乎 等 价 的 “文档 一 视图 结构 ?中 ， 大 家 可 以 看 到 这 个 问题 。 现 在 我 们 有 
一 些 数据 COCR") 以 及 多 个 视图 ， 假 定 为 一 张 图 (Plot) 和 一 个 文本 
视图 。 知 改变 了 数据 ， 两 个 视图 必须 知道 对 目 己 进行 更 新 ， 而 那 正 

是 “观察 器 ”要 负责 的 工作 。 这 是 一 种 十 分 常见 的 问题 ， 它 的 解决 方案 已 
包括 进 标准 的 java.util 库 中 。 


在 Java 中 ， 有 两 种 类 型 的 对 象 用 来 实现 观察 器 范式 。 其 中 ，Observable 
类 用 于 跟踪 那些 当 发 生 一 个 改变 时 希望 收 到 通知 的 所 有 个 体 无 
论 “ 状 态 ” 是 否 改 变 。 如 果 有 人 说 “好 了 ， 所 有 人 都 要 检查 自己 ， 并 可 能 
要 进行 更 新 ”， 那 么 Observable 类 会 执行 这 个 任务 为 列表 中 的 每 
个 “人 ”都 调用 notifyObservers() 方 法 。notifyObservers() 方 法 属于 基础 类 
Observable 的 一 部 分 。 


在 观察 器 范式 中 ， 实 际 有 两 个 方面 可 能 发 生变 化 : 观 罕 对 象 的 数量 以 及 
更 新 的 方式 。 也 就 是 说， 观察 恬 范式 允许 我 们 同时 修改 这 两 个 方面 ， 不 
会 干扰 围绕 在 它 周 围 的 其 他 代码 。 


下 面 这 个 例子 类 似 于 第 14 章 的 ColorBoxes 示 例 。 箱 子 (Boxes) 置 于 一 
个 屏幕 网 格 中 ， 每 个 都 初始 化 一 种 随机 的 颜色 。 此 外 ， 每 个 箱子 都 “ 实 
现 ”(implement) 了 “观察 器 ”(Observer) 接口 ， 而 且 随 一 个 Observable 
对 象 进 行 了 注册 。 知 点 击 一 个 箱子 ， 其 他 所 有 箱子 都 会 收 到 一 个 通知 ， 
指出 一 个 改变 已 经 发 生 。 这 是 由 于 Observable 对 象 会 目 动 调用 每 个 
Observer 对 象 的 update0) 方 法 。 在 这 个 方法 内 ， 箱 子 会 检查 被 点 中 的 那个 
箱子 是 否 与 自己 紧邻 。 若 答案 是 肯定 的 ， 那 么 也 修改 自己 的 颜色 ， 保 持 
与 点 中 那个 箱子 的 协调 。 


//: BoxObserver.java 


























// Demonstration of Observer pattern using 


// Java's built-in observer classes. 


import java.awt.*; 
import java.awt.event.*; 
import java.util.*; 
// You must inherit a new type of Observable: 
class BoxObservable extends Observable { 
public void notifyObservers(Object b) { 
// Otherwise it won't propagate changes: 
setChanged(); 


super .notifyObservers(b); 


J 


public class BoxObserver extends Frame { 
Observable notifier = new BoxObservable(); 
public BoxObserver(int grid) { 
setTitle("Demonstrates Observer pattern"); 
setLayout(new GridLayout(grid, grid)); 
for(int x = 0; x < grid; x++) 
for(int y = 0; y < grid; y++) 
add(new OCBox(x, y, notifier)); 
} 
public static void main(String[] args) { 
int grid = 8; 


if(args.length > 0) 


grid = Integer.parseInt(args[0]); 
Frame f = new BoxObserver(grid); 
f.setSize(500, 400); 
f.setVisible(true); 
f .addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


}); 


class OCBox extends Canvas implements Observer { 

Observable notifier; 

int x, y; // Locations in grid 

Color cColor = newColor(); 

static final Color[] colors = { 
Color.black, Color.blue, Color.cyan, 
Color.darkGray, Color.gray, Color.green, 
Color.lightGray, Color.magenta, 
Color.orange, Color.pink, Color.red, 


Color.white, Color.yellow 


}; 


static final Color newColor() { 
return colors[ 
(int)(Math.random() * colors.length) 
]; 
} 
OCBox(int x, int y, Observable notifier) { 


this.x = x; 


this.y y; 
notifier .addObserver (this); 
this.notifier = notifier; 
addMouseListener(new ML()); 

} 

public void paint(Graphics g) { 
g.setColor(cColor); 

Dimension s = getSize(); 
g.fillRect(0, ©, s.width, s.height); 

} 

class ML extends MouseAdapter { 


public void mousePressed(MouseEvent e) { 


notifier.notifyObservers(OCBox.this); 


} 


public void update(Observable o, Object arg) 


OCBox clicked = (OCBox)arg; 
if(nextTo(clicked)) { 
cColor = clicked.cColor; 


repaint(); 


} 
private final boolean nextTo(OCBox b) { 
return Math.abs(x - b.x) <= 1 && 
Math.abs(y - b.y) <= 1; 


} 
} ///:~ 


如 果 是 首次 查阅 Observable 的 联机 帮助 文档 ， 可 能 会 多 少 感 到 有 些 困 
惑 ， 因 为 它 似 乎 表明 可 以 用 一 个 原始 的 Observable 对 象 来 管理 更 新 。 但 
这 种 说 法 是 不 成 立 的 ， 大 家 可 自己 试 试 在 BoxObserver 中 ， 创 建 一 
个 Observable 对 象 ， 蔡 换 BoxObservable 对 象 ， 看 看 会 有 什么 事情 发 生 。 
事实 上 ， 什 么 事情 也 不 会 发 生 。 为 真正 产生 效果 ， 必 须 从 Observable 继 
承 ， 并 在 衍生 类 代码 的 某 个 地 方 调用 setChanged()。 这 个 方法 需要 设 
置 “changed”( 己 改变 ) 标志 ， 它 意味 着 当 我 们 调用 notifyObservers() 的 
时 候 ， 所 有 观察 器 事实 上 都 会 收 到 通知 。 在 上 面 的 例子 中 ， 
setChanged() 只 是 简单 地 在 notifyObservers() 中 调用 ， 大 家 可 依据 符合 实 
际 情况 的 任何 标准 决定 何 时 调用 setChanged()。 


BoxObserver 包 含 了 单个 Observable 对 象 ， 名 为 notifier。 每 次 创建 一 个 
OCBox 对 象 时 ， 它 都 会 同 notifier 联 系 到 一 起 。 在 OCBox 中 ， 只 要 点 击 鼠 
标 ， 就 会 发 出 对 notifyObservers(0) 方 法 的 调用 ， 并 将 被 点 中 的 那个 对 象 作 
为 一 个 参数 传递 进去 ， 使 收 到 消息 〈 用 它们 的 update(0) 方 法 ) 的 所 有 箱 
子 都 能 知道 谁 被 点 中 了 ， 并 据 此 判断 上 自己 是 否 也 要 变动 。 通 过 
notifyObserversO0 和 update0 中 的 代码 的 结合 ， 我 们 可 以 应 付 一 些 非 党 复 
杂 的 局 面 。 

















在 notifyObservers() 方 法 中 ， 表 面 上 似乎 观察 器 收 到 通知 的 方式 必须 在 编 
译 期 间 固 定 下 来 。 然 而 ， 只 要 稍微 仔细 研究 一 下 上 面 的 人 代码， 就 会 发 现 
BoxObserver 或 OCBox 中 唯一 需要 留意 是 否 使 用 BoxObservable 的 地 方 就 
是 创建 Observable 对 象 的 时 候 一 一 从 那 时 开始 ， 所 有 东西 都 会 使 用 基本 
的 Observable 接 口 。 这 意味 着 以 后 奉 想 更 改 通 知 方式 ， 可 以 继承 其 他 
Observable 类 ， 并 在 运行 期 间 交 换 它 们 。 





16.3 模拟 垃圾 回收 站 


这 个 问题 的 本 质 是 知 将 垃圾 丢 进 单个 垃圾 简 ， 事 实 上 是 未 经 分 类 的 。 但 
在 以 后 ， 某 些 特殊 的 信息 必须 恢复 ， 以 便 对 垃圾 正确 地 归 类 。 在 最 开始 
的 解决 方案 中 ，RTTI 扮 演 了 关键 的 角色 〔 详 见 第 11 章 )。 


这 并 不 是 一 种 普通 的 设计 ， 因 为 它 增加 了 一 个 新 的 限制 。 正 是 这 个 限制 
使 问题 变 得 非常 有 趣 一 一 它 更 象 我 们 在 工作 中 碰 到 的 那些 非常 豚 烦 的 问 
题 。 这 个 额外 的 限制 是 : 垃圾 抵达 垃圾 回收 站 时 ， 它 们 全 都 是 混合 在 一 
起 的 。 程 序 必须 为 那些 垃圾 的 分 类 定 出 一 个 模型 。 这 正 是 RTTI 发 挥 作 
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//: RecycleA. java 


// Recycling with RTTI 

package ci6.recyclea; 

import java.util.*; 

import java.io.*; 

abstract class Trash { 
private double weight; 
Trash(double wt) { weight = wt; } 
abstract double value(); 
double weight() { return weight; } 
// Sums the value of Trash in a bin: 

static void sumValue(Vector bin) { 

Enumeration e = bin.elements(); 


double val = 0.0f; 


while(e.hasMoreElements()) { 
// One kind of RTTI: 
// A dynamically-checked cast 
Trash t = (Trash)e.nextElement(); 
// Polymorphism in action: 
val += t.weight() * t.value(); 
System.out.printin( 
"weight of " + 
// Using RTTI to get type 
// information about the class: 
t.getClass().getName() + 
"=" + t.weight()); 
} 


System.out.printin("Total value = " + val); 


} 

class Aluminum extends Trash { 
static double val = 1.67f; 
Aluminum(double wt) { super(wt); } 
double value() { return val; } 
static void value(double newval) { 


val = newval; 


} 


class Paper extends Trash { 
static double val = 0.10f; 
Paper(double wt) { super(wt); } 
double value() { return val; } 
static void value(double newval) { 


val = newval; 


} 


class Glass extends Trash { 
static double val = 0.23f; 
Glass(double wt) { super(wt); } 
double value() { return val; } 
static void value(double newval) { 


val = newval; 


} 


public class RecycleA { 
public static void main(String[] args) { 
Vector bin = new Vector(); 
// Fill up the Trash bin: 
for(int i = 0; i < 30; i++) 


Switch((int)(Math.random() * 3)) { 


case 0 
bin.addElement (new 
Aluminum(Math.random() * 100)); 
break; 
case 1 
bin.addElement (new 
Paper(Math.random() * 100)); 
break; 
case 2 
bin.addElement (new 
Glass(Math.random() * 100)); 
} 
Vector 
glassBin = new Vector(), 
paperBin = new Vector(), 
alBin = new Vector(); 
Enumeration sorter = bin.elements(); 
// Sort the Trash: 
while(sorter.hasMoreElements()) { 
Object t = sorter.nextElement(); 
// RTTI to show class membership: 
if(t instanceof Aluminum) 


alBin.addElement(t); 


if(t instanceof Paper) 
paperBin.addElement(t); 
if(t instanceof Glass) 


glassBin.addElement(t); 


Trash.sumValue(alBin) ; 
Trash.sumValue(paperBin) ; 
Trash.sumValue(glassBin); 
Trash.sumValue(bin); 


} 
} ///:~ 


要 注意 的 第 一 个 地 方 是 package 语 句 : 
package c16.recyclea; 
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16 章 的 程序 ) 分 文 出 来 的 recyclea 子 目录 中 。 第 17 章 的 解 包工 具 会 负 贡 

将 其 置 入 正确 的 子 目 录 。 之 所 以 要 这 样 做 ， 是 因为 本 章 会 多 次 改写 这 个 

E EE 
和 冲突。 


其 中 创建 了 几 个 Vector 对 象 ， 用 于 容纳 Trash 人 句 顶 。 当 然 ，Vector 实 际 容 
纳 的 是 Object OTR) ， 所 以 它们 最 终 能 够 容纳 任何 东西 。 之 所 以 要 它 
们 容纳 Trash 〈 或 者 从 Trash 衍 生出 来 的 其 他 东西 ) ， 唯 一 的 理由 是 我 们 
需要 谨慎 地 避免 放 入 除 Trash 以 外 的 其 他 任何 东西 。 如 果真 的 把 某 些 “ 错 
误 ” 的 东西 置 入 Vector， 那 么 不 会 在 编译 期 得 到 出 错 或 警告 提示 只 能 
通过 运行 期 的 一 个 违例 知道 自己 已 经 犯 了 错误 。 


Trash 句 柄 加 入 后 ， 它 们 会 丢失 自己 的 特定 标识 信息 ， 只 会 成 为 简单 的 














Object 句柄 〈 上 漳 造 型 ) 。 然 而 ， 由 于 存在 多 形 性 的 因素 ， 所 以 在 我 们 
通过 Enumeration ”sorter 调 用 动态 绑 定 方法 时 ， 一 旦 结果 Object 已 经 造型 
回 Trash， 仍 然 会 发 生 正 确 的 行为 。sumValue0 也 用 一 个 Enumeration 对 
Vector 中 的 每 个 对 象 进行 操作 。 


表面 上 持 ， 先 把 Trash 的 类 型 上 漳 造 型 到 一 个 集合 容纳 基础 类 型 的 名 
柄 ， 再 回 过 头 重新 下 调 造 型 ， 这 似乎 是 一 种 非常 思 奏 的 做 法 。 为 什么 不 
只 是 一 开始 束 将 垃圾 置 入 适当 的 容器 里 呢 ? EKE, KEERA H 
收 ?一 团 迷 雾 的 关键 )。 在 这 个 程序 中 ， 我 们 很 容易 就 可 以 换 成 这 种 做 
但 在 茶 些 情况 下 ， 系 统 的 结构 及 灵活 性 都 能 从 下 调 造 型 中 得 到 极 大 
J 好 处 。 


该 程序 已 满足 了 设计 的 初 囊 ， 它 能 够 正常 工作 ! 只 要 这 是 个 一 次 性 的 方 
案 ， 就 会 显得 非常 出 色 。 但 是 ， 真 正 有 用 的 程序 应 该 能 够 在 任何 时 候 解 
决 问题 。 所 以 必须 间 自 己 这 样 一 个 问题 : “如 果 情 况 发 生 了 变化 ， 它 还 
能 工作 吗 ? ? 举 个 例子 来 说 ， 厚 纸 板 现在 是 一 种 非 党 有 价值 的 可 回收 物 
品 ， 那 么 如 何 把 它 集 成 到 系统 中 呢 《〈 特 别 是 程序 很 大 很 复杂 的 时 候 ) ? 
由 于 前 面 在 switch 语 句 中 的 类 型 检查 编码 可 能 散布 于 整个 程序 ， 
次 加 入 一 种 新 类 型 时 ， 都 必须 找到 所 有 那些 编码 。 知 不 慎 遗 漏 一 

译 器 除了 指出 存在 一 个 错误 之 外 ， 不 能 再 提供 任何 有 价值 的 帮助 。 


RTITI 在 这 里 使 用 不 当 的 关键 是 “每 种 类 型 都 进行 了 测试 >。 如 果 由 于 类 型 
的 子 集 需 要 特殊 的 对 符 ， 所 以 只 寻找 那个 子 集 ， 那 么 情况 就 会 变 得 好 一 
些 。 但 假如 在 一 个 switch 语 句 中 查找 每 一 种 类 型 ， 那 么 很 可 能 错过 一 个 
重点 ， 使 最 终 的 代码 很 难 维护 。 在 下 一 节 中 ， 大 家 会 学 习 如 何 逐 步 对 这 
使 其 显得 越 来 越 郝 活 。 这 是 在 程序 设计 中 一 种 非常 有 
意义 的 侦 















































16.4 改进 议 计 


(Design Patters》 书 内 所 有 方案 的 组 织 都 围绕 “程序 进化 时 会 发 生 什 么 

变化 "这 个 问题 展开 。 对 于 任何 设计 来 说 ， 这 都 可 能 是 最 重要 的 一 个 问 
题 。 大 根据 对 这 个 问题 的 回答 来 构造 自己 的 系统 ， 束 可 以 得 到 两 个 方面 
的 结果 : 系统 不 仅 更 易 维 护 《〈 而 且 更 廉价 ) ， 而 且 能 产生 一 些 能 够 重复 
使 用 的 对 象 ， 进 而 使 其 他 相关 系统 的 构造 也 变 得 更 廉价 。 这 正 是 面 癌 对 
象 程 序 设 计 的 优势 所 在 ， 但 这 一 优势 并 不 是 自动 体现 出 来 的 。 它 要 求 对 
我 们 对 需要 解雇 的 问题 有 全 面 而 且 深 入 的 理解 。 在 这 一 节 中 ， 我 们 准备 
在 系统 的 逐步 改进 过 程 中 同 大 家 展示 如 何 做 到 这 一 点 。 


就 目前 这 个 回收 系统 来 说 ， 对 “什么 会 变化 "这 个 问题 的 回答 是 非常 普通 
的 : 更 多 的 类 型 会 加 入 系统 。 因 此 ， 设 计 的 目标 就 是 尽 可 能 简化 这 种 类 
型 的 添加 。 在 回收 程序 中 ， 我 们 准备 把 涉及 特定 类 型 信息 的 所 有 地 方 都 
封装 起 来 。 这 样 一 来 (如 果 没 有 别 的 原因 〉 ， 所 有 变化 对 那些 封装 来 说 
都 是 在 本 地 进行 的 。 这 种 处 理 方式 也 使 代码 剩余 的 部 分 显得 特别 清爽 。 


16.4.1 “制作 更 多 的 对 象 ” 


这 样 便 引 出 了 面向 对 象 程序 设计 时 一 条 常规 的 准则 ， 我 最 早 是 在 Grady 
Booch EWW: ARITE, WEEE WNR”. AEE 
来 有 些 暧昧 ， 且 简单 得 可 笑 ， 但 这 确实 是 我 知道 的 最 有 用 一 条 准则 《大 
家 以 后 会 注意 到 “制作 更 多 的 对 象 ” 经 常 等同 于 “添加 为 一 个 层次 的 迁 
回 ”) 。 一般 情况 下 ， 如 果 发 现 一 个 地 方 充斥 着 大 量 繁复 的 代码 ， 束 需 
要 考虑 什么 类 能 使 它 显 得 清 碍 一 些 。 用 这 种 方式 整理 系统 ， 往 往 会 得 到 
一 个 更 好 的 结构 ， 也 使 程序 更 加 灵活 。 


首先 考虑 Trash 对 象 首次 创建 的 地 方 ， 这 是 main0 里 的 一 个 switch 语 句 : 


for(int i = 0; i < 30; i++) 























switch((int)(Math.random() * 3)) { 
case 0 : 


bin.addElement (new 


Aluminum(Math.random() * 100)); 
break; 
case 1: 
bin.addElement (new 
Paper(Math.random() * 100)); 
break; 
case 2 : 
bin.addElement (new 


Glass(Math.random() * 100)); 





这 些 代 码 显 然 * 过 于 复杂 ”， 也 是 新 类 型 加 入 时 必须 改动 代码 的 场所 之 
一 。 如 果 经 党 都 要 加 入 新 类 型 ， 那 么 更 好 的 方案 就 是 建立 一 个 独立 的 方 
法 ， 用 它 获 取 所 有 必需 的 信息 ， 并 创建 一 个 句柄 ， 指 问 正 确 类 型 的 一 个 
对 象 一 一 已 经 上 渊 造型 到 一 个 Trash 对 象 。 在 《Design Patterns) F, © 
被 粗略 地 称呼 为 “创建 范式 ”。 要 在 这 里 应 用 的 特殊 范式 是 Factory 方 法 的 
一 种 变 体 。 在 这 里 ，Factory 方 法 属于 Trash 的 一 名 static《〈 静 态 ) 成 员 。 
但 更 常见 的 一 种 情况 是 : 它 属于 衍生 类 中 一 个 被 过 载 的 方法 。 


Factory 方 法 的 基本 原理 是 我 们 将 创建 对 象 所 需 的 基本 信息 传递 给 它 ， 然 
后 返回 并 等 候 句 柄 《已 经 上 漳 造 型 至 基础 类 型 ) 作为 返回 值 出 现 。 从 这 
时 开始 ， 就 可 以 按 多 形 性 的 方式 对 待 对象 了 。 因 此 ， 我 们 根本 没 必要 知 
道 所 创建 对 象 的 准确 类 型 是 什么 。 事 实 上 ，Factory 方 法 会 把 自己 隐藏 起 
来 ， 我 们 是 看 不 见 它 的 。 这 样 做 可 防止 不 愤 的 误 用 。 如 果 想 在 没有 多 形 
性 的 前 提 下 使 用 对 象 ， 必 须 明确 地 使 用 RTTI 和 指定 造型 。 


但 仍然 存在 一 个 小 问题 ， 特 别 是 在 基础 类 中 使 用 更 复杂 的 方法 〈 不 是 在 
这 里 展示 的 那 种 ) ， 且 在 衍生 类 里 过 载 〈 履 盖 ) 了 它 的 前 提 下 。 如 果 在 
衍生 类 里 请 求 的 信息 要 求 更 多 或 者 不 同 的 参数 ， 那 么 该 怎么 办 呢 ? “Bl 
建 更 多 的 对 象 " 解 决 了 这 个 问题 。 为 实现 Factory 方 法 ，Trash 类 使 用 了 一 
个 新 的 方法 ， 名 为 factory。 为 了 将 创建 数据 隐藏 起 来 ， 我 们 用 一 个 名 为 














Info 的 新 类 包含 factory 方 法 创建 适当 的 Trash 对 象 时 需要 的 全 部 信息 。 下 
面 是 mfo 一 种 简单 的 实现 方式 : 


class Info { 


int type; 
// Must change this to add another type: 
static final int MAX_NUM = 4; 
double data; 
Info(int typeNum, double dat) { 
type = typeNum % MAX_NUM; 


data = dat; 


Info 对 象 唯 一 的 任务 就 是 容纳 用 于 factory() 方 法 的 信息 。 现 在 ， 假 如 出 现 
了 一 种 特殊 情况 ，factory0 需 要 更 多 或 者 不 同 的 信息 来 新 建 一 种 类 型 的 
Trash 对 象 ， 那 么 再 也 不 需要 改动 factory() 了 。 通 过 添加 新 的 数据 和 构建 
器 ， 我 们 可 以 修改 Info 类 ， 或 者 采用 子 类 处 理 更 典型 的 面 癌 对 象形 式 。 


用 于 这 个 简单 示例 的 factory0 方 法 如 下 : 





static Trash factory(Info i) { 


Switch(i.type) { 
default: // To quiet the compiler 
case 0: 


return new Aluminum(i.data); 


case 1: 
return new Paper(i.data); 
case 2: 
return new Glass(i.data); 
// Two lines here: 
case 3: 


return new Cardboard(i.data); 


在 这 里 ， 对 象 的 准确 类 型 很 容易 即 可 判断 出 来 。 但 我 们 可 以 设想 一 些 更 
复杂 的 情况 ，factory0 将 采用 一 种 复杂 的 算法 。 无 论 如 何 ， 现 在 的 关键 
是 它 已 隐藏 到 某 个 地 方 ， 而 且 我 们 在 添加 新 类 型 时 知道 去 那个 地 方 。 
新 对 象 在 main0 中 的 创建 现在 变 得 非常 简单 和 清爽 : 


for(int i = 0; i < 30; i++) 








bin.addElement ( 
Trash. factory( 
new Info( 
(int)(Math.random() * Info.MAX_NUM), 


Math.random() * 100))); 


我 们 在 这 里 创建 了 一 个 Info 对 象 ， 用 于 将 数据 传 入 factory0; 后 者 在 内 存 
堆 中 创建 菜 种 Trash 对 象 ， 并 返回 添加 到 Vector bin 内 的 句柄 。 当 然 ， 如 
果 改 变 了 参数 的 数量 及 类 型 ， 仍 然 需要 修改 这 个 语句 。 但 假如 Info 对 象 
的 创建 是 目 动 进行 的 ， 也 可 以 避免 那个 腑 烦 。 例 如 ， 可 将 参数 的 一 个 


Vector 传 弟 到 Info 对 象 的 构建 器 中 (或 直接 传 入 一 个 factory() 调 用 ) 。 这 
要 求 在 运行 期 间 对 参数 〈 自 变量 ) 进行 分 析 与 检查 ， 但 确实 提供 了 非常 
高 的 灵活 程度 。 


大 家 从 这 个 代码 可 看 出 Factory 要 负责 解决 的 “领头 变化 ”问题 ， 如 果 问 系 
统 添加 了 新 类 型 (发 生 了 变化 ) ， 唯 一 需要 修改 的 代码 在 Factory 内 部 ， 
所 以 Factory 将 那 种 变化 的 影响 隔离 出 来 了 。 


16.4.2 用 于 原型 创建 的 一 个 范式 


上 述 设计 方案 的 一 个 问题 是 仍然 需要 一 个 中 心 场 所 ， 必 须 在 那里 知道 所 
有 类 型 的 对 象 : 在 factory0) 方 法 内 部 。 如 果 经 常 都 要 加 系统 添加 新 类 
型 ，factory(0 方 法 为 每 种 新 类 型 都 要 修改 一 届 。 各 确实 对 这 个 问题 感到 
否 恼 ， 可 试 试 再 深入 一 步 ， 将 与 类 型 有 关 的 所 有 信息 一 一 包括 它 的 创建 
过 程 一 一 都 移入 代表 那 种 类 型 的 类 内 部 。 这 样 一 来 ， 每 次 新 添 一 种 类 型 
的 时 候 ， 需 要 做 的 唯一 事情 就 是 从 一 个 类 继承 。 


为 将 涉及 类 型 创建 的 信息 移入 特定 类 型 的 Trash 里 ， 必 须 使 用 “ 原 

型 ”(prototype) yer (SEH (Design Patterns) AAS) 。 这 里 最 基本 
的 想法 是 我 们 有 一 个 主 控 对 象 序 列 ， 为 自己 感 兴趣 的 每 种 类 型 都 制作 一 
个 。 这 个 序列 中 的 对 象 只 能 用 于 新 对 象 的 创建 ， 采 用 的 操作 类 似 内 建 到 
Java 根 类 Object 内 部 的 cloneO 机 制 。 在 这 种 情况 下 ， 我 们 将 克隆 方法 命 
名 为 tClone()。 准 备 创建 一 个 新 对 象 时 ， 要 事先 收集 好 某 种 形式 的 信 
息 ， 用 它 建 立 我 们 希望 的 对 象 类 型 。 然 后 在 主 控 序 列 中 人 裔 历 ， 将 手 上 的 
信息 与 主 探 序列 中 原型 对 象 内 任何 适当 的 信息 作对 比 。 知 找到 一 个 符合 
自己 需要 的 ， 就 元 隆 它 。 


采用 这 种 方案 ， 我 们 不 必用 便 编 码 的 方式 植 入 任何 创建 信息 。 每 个 对 象 
都 知道 如 何 揭示 出 适当 的 信息 ， 以 及 如 何 对 上 自身 进行 克隆 。 所 以 一 种 新 
类 型 加 入 系统 的 时 候 ，factory(0) 方 法 不 需要 任何 改变 。 


为 解决 原型 的 创建 问题 ， 一 个 方法 是 添加 大 量 方 法 ， 用 它们 支持 新 对 象 
的 创建 。 但 在 Java ”1.1 中 ， 如 果 拥 有 指 癌 Class 对 象 的 一 个 句柄 ， 那 么 它 
己 经 提供 了 对 创建 新 对 象 的 支持 。 利 用 Java 1.1 的 “反射 > 〈 已 在 第 11 章 介 
绍 ) 技术 ， 即 便 我 们 只 有 指 癌 Class 对 象 的 一 个 句柄 ， 亦 可 正常 地 调用 一 
个 构建 器 。 这 对 原型 问题 的 解决 无 疑 是 个 完美 的 方案 。 


原型 列表 将 由 指向 所 有 想 创建 的 Class 对 象 的 一 个 句柄 列表 间接 地 表示 。 




































































除 此 之 外 ， 假 如 原型 处 理 失 败 ， 则 factory() 方 法 会 认为 由 于 一 个 特定 的 

Class 对 象 不 在 列表 中 ， 所 以 会 答 试 装载 它 。 通 过 以 这 种 方式 动态 装载 原 
型 ，Trash 类 根本 不 需要 知道 自己 要 操纵 的 是 什么 类 型 。 因 此 ， 在 我 们 

添加 新 类 型 时 不 需要 作出 任何 形式 的 修改 。 于 是 ， 我 们 可 在 本 章 剩 余 的 
部 分 方便 地 重复 利用 它 。 


//: Trash.java 








// Base class for Trash recycling examples 
package c1i6.trash; 
import java.util.*; 
import java.lang.reflect.*; 
public abstract class Trash { 
private double weight; 
Trash(double wt) { weight = wt; } 
Trash() {} 
public abstract double value(); 
public double weight() { return weight; } 
// Sums the value of Trash in a bin: 
public static void sumValue(Vector bin) { 
Enumeration e = bin.elements(); 
double val = 0.0f; 
while(e.hasMoreElements()) { 
// One kind of RTTI: 
// A dynamically-checked cast 


Trash t = (Trash)e.nextElement(); 


val += t.weight() * t.value(); 
System. out.printin( 
"weight of " + 
// Using RTTI to get type 
// information about the class: 
t.getClass().getName() + 
"=" + t.weight()); 


} 


System.out.printin("Total value = " + val); 
} 
// Remainder of class provides support for 
// prototyping: 
public static class PrototypeNotFoundException 
extends Exception {} 
public static class CannotCreateTrashException 
extends Exception {} 
private static Vector trashTypes = 
new Vector(); 
public static Trash factory(Info info) 
throws PrototypeNotFoundException, 
CannotCreateTrashException { 
for(int i = 0; i < trashTypes.size(); i++) { 


// Somehow determine the new type 


// to create, and create one: 
Class tc = 
(Class)trashTypes.elementAt(1); 
if (tc.getName().indexOf(info.id) != -1) { 
try { 
// Get the dynamic constructor method 
// that takes a double argument: 
Constructor ctor = 
tc.getConstructor ( 
new Class[] {double.class}); 
// Call the constructor to create a 
// new object: 
return (Trash)ctor.newInstance( 
new Object[]{new Double(info.data)}); 
} catch(Exception ex) { 
ex.printStackTrace(); 


throw new CannotCreateTrashException(); 


} 


// Class was not in the list. Try to load it, 


// but it must be in your class path! 


try { 


System.out.printin("Loading " + info.id); 
trashTypes.addElement ( 
Class.forName(info.id)); 
} catch(Exception e) { 
e.printStackTrace(); 
throw new PrototypeNotFoundException(); 
} 
// Loaded successfully. Recursive call 
// should work this time: 
return factory(info); 
} 
public static class Info { 
public String id; 
public double data; 
public Info(String name, double data) { 
id = name; 


this.data = data; 


} 
二 


基本 Trash 类 和 sumValue0 还 是 象 往常 一 样 。 这 个 类 剩 下 的 部 分 支持 原型 
范式 。 大 家 首先 会 看 到 两 个 内 部 类 〈 被 设 为 static 属 性 ， 使 其 成 为 只 为 代 
码 组 织 目的 而 存在 的 内 部 类 ) ， 它 们 摘 述 了 可 能 出 现 的 违例 。 在 它 后 面 








跟随 的 是 一 个 Vector trashTypes， 用 于 容纳 Class 句 柄 。 


在 Trash.factory() 中 ，Info 对 象 id (Info 类 的 男 一 个 版 本 ， 与 前 面 讨论 的 不 
HD 内 部 的 String 包 含 了 要 创建 的 那 种 Trash 的 类 型 名 称 。 这 个 String 会 与 
列表 中 的 Class 名 比较 。 硅 存在 相符 的 ， 那 便 是 要 创建 的 对 象 。 当 然 ， 还 
有 很 多 方法 可 以 决定 我 们 想 创 建 的 对 象 。 之 所 以 要 采用 这 种 方法 ， 是 因 
为 从 一 个 文件 读 入 的 信息 可 以 转换 成 对 象 。 


RIA CAEN Trash (垃圾 ) 种 类 后 ， 接 下 来 就 轮 到 “反射 ”方法 大 显 
身手 了 。getConstructor() 方 法 需要 取得 自己 的 参数 一 一 由 Class 人 句柄 构成 
的 一 个 数组 。 这 个 数组 代表 着 不 同 的 参数 ， 并 按 它们 正确 的 顺序 排列 ， 

以 便 我 们 查找 的 构建 器 使 用 。 在 这 儿 ， 该 数组 是 用 Java 1.1 的 数组 创建 
语法 动态 创建 的 : 








new Class[] {double.class} 


这 个 代码 假定 所 有 Trash 类 型 都 有 一 个 需要 double 数 值 的 构建 器 (注意 
double.class 与 Double.class 是 不 同 的 ) 。 若 考虑 一 种 更 灵活 的 方案 ， 亦 可 
调用 getConstructors()， 令 其 返回 可 用 构建 器 的 一 个 数组 。 


从 getConstructors0 返 回 的 是 指 同一 个 Constructor 对 象 的 句柄 〈 该 对 象 是 
java.lang.reflect 的 一 部 分 ) 。 我 们 用 方法 newInstance0 动 态 地 调用 构建 
器 。 该 方法 需要 获取 包含 了 实际 参数 的 一 个 Object 数组 。 这 个 数组 同样 
是 按 Java 1.1 的 语法 创建 的 : 





new Object[] {new Double(info.data)} 


在 这 种 情况 下 ，double 必 须 置 入 一 个 封装 〈 容 器 ) 类 的 内 部 ， 使 其 真正 
成 为 这 个 对 象 数 组 的 一 部 分 。 通 过 调用 newInstance()， 会 提取 出 
double， 但 大 家 可 能 会 党 得 稍微 有 些 迷 惑 一 一 参数 既 可 能 是 double， 也 
可 能 是 Double， 但 在 调用 的 时 候 必 须 用 Double 传 递 。 幸 运 的 是 ， 这 个 问 
题 只 存在 于 基本 数据 类 型 中 间 。 


理解 了 具体 的 过 程 后 ， 再 来 创建 一 个 新 对 象 ， 并 且 只 为 它 提供 一 个 Class 
人 句柄， 事情 就 变 得 非常 简单 了 。 束 目前 的 情况 来 说 ， 内 部 循环 中 的 
return 水 远 不 会 执行 ， 我 们 在 终点 就 会 退出 。 在 这 儿 ， 程 序 动态 装载 
Class 对 象 ， 并 把 它 加 入 trashTypes 《垃圾 类 型 ) 列表， 从 而 试图 纠正 这 
个 问题 。 硅 仍然 找 不 到 真正 有 问题 的 地 方 ， 同 时 闭 载 义 是 成 功 的 ， 那 么 





























MER HH factory ik, EINA. 

正如 大 家 会 看 到 的 那样 ， 这 种 设计 方案 最 大 的 优点 就 是 不 需要 改动 代 
码 。 无 论 在 什么 情况 下 ， 它 都 能 正常 地 使 用 (假定 所 有 Trash 子 类 都 包 
售 了 一 个 构建 问 ， 用 以 获取 单个 double 参 数 ) 。 

1. Trash 子 类 


为 了 与 原型 机 制 相 适应 ， 对 Trash 每 个 新 子 类 唯一 的 要 求 就 是 在 其 中 包 
含 了 一 个 构建 器 ， 指 示 它 获取 一 个 double 参 数 。Java 1.1 的 “反射 ?机 制 可 
负 贡 剩 下 的 所 有 工作 。 

下 面 是 不 同类 型 的 Trash， 每 种 类 型 都 有 它们 自己 的 文件 里 ， 但 都 属于 
Trash 包 的 一 部 分 (同样 地 ， 为 了 方便 在 本 间 内 重复 使 用 ) : 


//: Aluminum. java 








// The Aluminum class with prototyping 

package c16.trash; 

public class Aluminum extends Trash { 
private static double val = 1.67f; 
public Aluminum(double wt) { super(wt); } 
public double value() { return val; } 
public static void value(double newVal) { 

val = newVal; 

} 

} ///:~ 


下 面 是 一 种 新 的 Trash 类 型 : 


//: Cardboard.java 


// The Cardboard class with prototyping 

package ci6.trash; 

public class Cardboard extends Trash { 
private static double val = 0.23f; 
public Cardboard(double wt) { sSuper(wt); } 
public double value() { return val; } 
public static void value(double newVal) { 


val = newVal; 


} ///:~ 


可 以 看 出 ， 除 构建 右 以 外 ， 这 些 类 根本 没有 什么 特别 的 地 方 。 
2. 从 外 部 文件 中 解析 出 Trash 
与 Trash 对 象 有 关 的 信息 将 从 一 个 外 部 文件 中 读 取 。 人 针对 Trash 的 每 个 方 


面 ， 文 件 内 列 出 了 所 有 必要 的 信息 一 一 每 行 都 代表 一 个 方面 ， 采 用 "“ 垃 
圾 《废品 ) 名 称 : 值 ?的 固定 格式 。 例 如 : 


ci6.Trash.Glass:54 








c16.Trash.Paper:22 
c16.Trash.Paper:11 
c16.Trash.Glass:17 
c16.Trash.Aluminum: 89 


ci6.Trash.Paper:88 


c16. 


c16. 


c16. 


c16. 


c16 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


c16. 


Trash. 


Trash. 


Trash. 


Trash. 


.Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Trash. 


Aluminum:76 
Cardboard:96 
Aluminum: 25 
Aluminum: 34 
Glass:11 
Glass:68 
Glass:43 
Aluminum: 27 
Cardboard: 44 
Aluminum: 18 
Paper :91 
Glass:63 
Glass:50 
Glass: 80 
Aluminum: 81 
Cardboard:12 
Glass:12 
Glass:54 
Aluminum: 36 
Aluminum: 93 
Glass:93 
Paper :80 


Glass:36 


c16.Trash.Glass:12 
c16.Trash.Glass:60 
c16.Trash.Paper : 66 
c16.Trash.Aluminum: 36 


ci6.Trash.Cardboard: 22 





注意 在 给 定 类 名 的 时 候 ， 类 路 径 必 须 包 含 在 内 ， 人 否则 就 找 不 到 类 。 


为 解析 它 ， 每 一 行内 容 都 会 谈 入 ， 并 用 字 串 方法 indexOfO 来 建立 “的 一 
个 索引 。 首 移 用 字 串 方法 substring0 取 出 世 圾 的 类 型 名 称 ， 接 着 用 一 个 
静态 方法 Double.valueOf() 取 得 相应 的 值 ， 并 转换 成 一 个 double 值 。trim() 
方法 则 用 于 删除 字 串 两 头 的 多 余 空 格 。 


Trash 解 析 器 置 入 单独 的 文件 中 ， 因 为 本 章 将 不 断 地 用 到 它 。 如 下 所 
ZN: 








//: ParseTrash.java 


// Open a file and parse its contents into 
// Trash objects, placing each into a Vector 
package c16.trash; 
import java.util.*; 
import java.io.*; 
public class ParseTrash { 

public static void 

fillBin(String filename, Fillable bin) { 


try { 


BufferedReader data = 


new BufferedReader ( 
new FileReader(filename) ) ) 
String buf; 
while((buf = data.readLine())!= null) { 
String type = buf.substring(0, 
buf .indexOf(':')).trim(); 
double weight = Double.valueOf ( 
buf .substring(buf.indexOf(':') + 1) 
.trim()).doubleValue(); 
bin.addTrash( 
Trash. factory( 
new Trash.Info(type, weight))); 
} 
data.close(); 
} catch(IOException e) { 
e.printStackTrace(); 
} catch(Exception e) { 


e.printStackTrace(); 


} 


// Special case to handle Vector: 
public static void 


fillBin(String filename, Vector bin) { 


fillBin(filename, new FillableVector(bin)); 


Í 
} ///:~ 


在 RecycleA.java 中 ， 我 们 用 一 个 Vector 容纳 Trash 对 象 。 然 而 ， 亦 可 考虑 
采用 其 他 集合 类 型 。 为 做 到 这 一 点 ，flBin0 的 第 一 个 版 本 将 获取 指 辐 
人 
方法 : 


//: Fillable.java 


// Any object that can be filled with Trash 
package ci6.trash; 
public interface Fillable { 

void addTrash(Trash t); 


} ///:~ 


支持 该 接口 的 所 有 东西 都 能 伴随 fillBin 使 用 。 当 然 ，Vector 并 未 实现 
Fillable， 所 以 它 不 能 工作 。 由 于 Vector 将 在 大 多 数 例子 中 应 用 ， 所 以 最 
好 的 做 法 是 添加 另 一 个 过 载 的 名 Bin0 方 法 ， 令 其 以 一 个 Vector 作为 参 
数 。 利 用 一 个 适配器 (Adapter) 类 ， 这 个 Vector 可 作为 一 个 Fillable 对 象 
使 用 : 


//: FillableVector.java 


// Adapter that makes a Vector Fillable 
package ci6.trash; 
import java.util.*; 


public class FillableVector implements Fillable { 


private Vector v; 

public FillableVector(Vector vv) { v = vv; } 

public void addTrash(Trash t) { 
v.addElement(t); 


} 
} ///:~ 


可 以 看 到 ， 这 个 类 唯一 的 任务 就 是 负责 将 Filable 的 addTrashO 同 Vector 的 
addFlement() 方 法 连接 起 来 。 利 用 这 个 类 ， 已 过 载 的 倍 Bin() 方 法 可 在 
ParseTrash.java 中 伴随 一 个 Vector 使 用 : 


public static void 


fillBin(String filename, Vector bin) { 


fillBin(filename, new FillableVector(bin)); 


这 种 方案 适用 于 任何 频繁 用 到 的 集合 类 。 除 此 以 外 ， 集 合 类 还 可 提供 它 
自己 的 适 配 右 类 ， 并 实现 Fillable 〈( 稍 后 即 可 看 到 ， 在 DynaTrash.java 
HA) 

3. 原型 机 制 的 重复 应 用 


现在 ， 大 家 可 以 看 到 采用 原型 技术 的 、 修 订 过 的 RecycleA.java 版 本 了 : 


//: RecycleAP.java 


// Recycling with RTTI and Prototypes 


package c16.recycleap; 


import c16.trash.*; 


import java.util.*; 


public class RecycleAP { 


public static void main(String[] args) 


Vector bin 


// Fill up 


ParsetTrash. 


Vector 
glassBin 
paperBin 


alBin = 


= new Vector(); 
the Trash bin: 


fillBin("Trash.dat", 


new Vector(), 


new Vector(), 


new Vector(); 


bin); 


Enumeration sorter = bin.elements(); 


// Sort the Trash: 


while(sorter.hasMoreElements()) { 


Object t 


= sorter.nextElement(); 


// RTTI to show class membership: 


if(t instanceof Aluminum) 


alBin.addElement(t); 


if(t instanceof Paper) 


paperBin.addElement(t); 


if(t instanceof Glass) 


glassBin.addElement(t); 


Trash.sumValue(alBin); 
Trash.sumValue(paperBin) ; 
Trash.sumValue(glassBin) ; 
Trash.sumValue(bin); 


} 
} ///:~ 


所 有 Trash 对 象 一 一 以 及 ParseTrash 及 支撑 类 现在 都 成 为 名 为 
cl6.trash 的 一 个 包 的 一 部 分 ， 所 以 它们 可 以 简单 地 导入 。 


无 论 打开 包含 了 Trash 摘 述 信息 的 数据 文件 ， 还 是 对 那个 文件 进行 解 
析 ， 所 有 涉及 到 的 操作 均 已 封装 到 static (静态 ) 方法 ParseTrash.fillBin() 
里 。 所 以 它 现在 已 经 不 是 我 们 设计 过 程 中 要 注意 的 一 个 重点 。 在 本 草 剩 
余 的 部 分 ， 大 家 经 常 都 会 看 到 无 论 添加 的 是 什么 类 型 的 新 类 ， 
ParseTrash.fillBin() 都 会 持续 工作 ， 不 会 发 生 改变 ， 这 无 疑 是 一 种 优 民 的 


设计 方案 。 


提 到 对 象 的 创建 ， 这 一 方案 确实 已 将 新 类 型 加 入 系统 所 需 的 变动 严格 
地 “本 地 化 ”了 。 但 在 使 用 RTTI 的 过 程 中 ， 却 存在 着 一 个 严重 的 问题 ， 这 
里 已 明确 地 显露 出 来 。 程 序 表面 上 工作 得 很 好 ， 但 却 永 远 侦 测 到 不 

能 “ 硬 纸 板 ”(Cardboard) 这 种 新 的 废品 类 型 即使 列表 里 确实 有 一 个 
便 纸板 类 型 ! 之 所 以 会 出 现 这 种 情况 ， 完 全 是 由 于 使 用 了 RTTI 的 缘 
故 。RTTI 只 会 查找 那些 我 们 告诉 它 查 找 的 东西 。RTTI 在 这 里 错误 的 用 
法 是 “系统 中 的 每 种 类 型 ”都 进行 了 测试 ， 而 不 是 仅 测 试 一 种 类 型 或 者 一 
个 类 型 子 集 。 正 如 大 家 以 后 会 看 到 的 那样 ， 在 测试 每 一 种 类 型 时 可 换 用 
其 他 方式 来 运用 多 形 性 特征 。 但 假如 以 这 种 形式 过 多 地 使 用 RTTI， 而 
旦 又 在 自己 的 系统 里 添加 了 一 种 新 类 型 ， 很 容易 就 会 在 记 在 程序 里 作出 
适当 的 改动 ， 从 而 埋 下 以 后 难以 发 现 的 Bug。 因 此 ， 在 这 种 情况 下 避免 
使 用 RTTI 是 很 有 必要 的 ， 这 并 不 仅仅 是 为 了 表面 好 看 一 一 也 是 为 了 产 
生 更 易 维 护 的 代码 。 


























16.5 抽象 的 应 用 


走 到 这 一 步 ， 接 下 来 该 考 碟 一 下 设计 方案 剩 下 的 部 分 了 一 一 在 哪里 使 用 
R? 既然 归 类 到 垃圾 箱 的 办 法 非常 不 雅 且 过 于 暴露 ， 为 什么 不 隔离 那个 
过 程 ， 把 它 隐 藏 到 一 个 类 里 呢 ? 这 就 是 著名 的 “如 采 必 须 做 不 雅 的 事 
情 ， 至 少 应 将 其 本 地 化 到 一 个 类 里 ”规则 。 看 起 来 残 象 下 面 这 样 : 


TrashSorter . 
Aluminum ArrayList 
Trash eine + 
Fees Paper ArrayList 
Glass ArrayList 


现在 ， 只 要 一 种 新 类 型 的 Trash 加 入 方法 ， 对 TrashSorter 对 象 的 初始 化 就 
必须 变动 。 可 以 想象 ，TrashSorter 类 看 起 来 应 该 象 下 面 这 个 样子 : 












class TrashSorter extends Vector { 
void sort(Trash t) { /* ... */ } 
} 


Eii, TrashSorter H— 44 A Vector (系列 ) ， 而 那些 
句柄 指向 的 又 是 由 Trash 句 柄 构成 的 Vector; 利用 addElementO0， 可 以 安 
装 新 的 TrashSorter， 如 下 所 示 : 


TrashSorter ts = new TrashSorter(); 
ts.addElement(new Vector()); 


但 是 现在 ，sort() 却 成 为 一 个 问题 。 用 静态 方式 编码 的 方法 如 何 应 付 一 种 
新 类 型 加 入 的 事实 呢 ? 为 解决 这 个 问题 ， 必 须 从 sortO 里 将 类 型 信息 删 

除 ， 使 其 需要 做 的 所 有 事情 就 是 调用 一 个 通用 方法 ， 用 它 照 料 涉及 类 型 
处 理 的 所 有 细节 。 这 当然 是 对 一 个 动态 绑 定 方法 进行 描述 的 另 一 种 方 

式 。 所 以 sort0 会 在 序列 中 简单 地 授 历 ， 并 为 每 个 Vector 都 调用 一 个 动态 
绑 定 方法 。 由 于 这 个 方法 的 任务 是 收集 它 感 兴趣 的 垃圾 片 ， 所 以 称 之 为 
grab(Trash)。 结 构 现 在 变 成 了 下 面 这 样 : 







Aluminum ArrayList 


boolean grab(Trash) 


Paper ArrayList 


boolean grab(Trash) 


Glass ArrayList 


boolean grab(Trash) 










TrashSorter 


ArrayList of J | 
Trash Bins 4 | 









其 中 ，TrashSorter 需 要 调用 每 个 grab() 方 法 ; 然后 根据 当前 Vector 容纳 的 
是 什么 类 型 ， 会 获得 一 个 不 同 的 结果 。 也 就 是 说 ，Vector 必 须 留意 目 己 
容纳 的 类 型 。 解 决 这 个 问题 的 传统 方法 是 创建 一 个 基础 “Trash bin” (hz 
RA) 类 ， 并 为 希望 容纳 的 每 个 不 同 的 类 型 都 继承 一 个 新 的 衍生 类 。 知 
Java 有 一 个 参数 化 的 类 型 机 制 ， 那 就 也 许 是 最 直接 的 方法 。 但 对 于 这 种 
机 制 应 该 为 我 们 构建 的 各 个 类 ， 我 们 不 应 该 进行 麻烦 的 手工 编码 ， 以 后 
的 “观察 ”方式 提供 了 一 种 更 好 的 编码 方式 。 


OOP 设 计 一 条 基本 的 准则 是 “为 状态 的 变化 使 用 数据 成 员 ， 为 行为 的 变 
化 使 用 多 性 形 ”。 对 于 容纳 Paper CRIK) 的 Vector， 以 及 容纳 Glass (IK 
璃 ) 的 Vector， 大 家 最 开始 或 许 会 认为 分 别 用 于 它们 的 grab0 方 法 肯定 会 
产生 不 同 的 行为 。 但 具体 如 何 却 完全 取决 于 类 型 ， 而 不 是 其 他 什么 东 
西 。 可 将 其 解释 成 一 种 不 同 的 状态 ， 而 且 由 于 Java 有 一 个 类 可 表示 类 型 
(Class) ， 所 以 可 用 它 判 断 特 定 的 Tbin 要 容纳 什么 类 型 的 Trash。 

用 于 Tbin 的 构建 器 要 求 我 们 为 其 传递 自己 选择 的 一 个 Class。 这 样 做 可 告 
诉 Vector 它 希望 容纳 的 是 什么 类 型 。 随 后 ，grab(0) 方 法 用 Class BinType 和 
RTTI 来 检查 我 们 传递 给 它 的 Trash 对 象 是 否 与 它 希 望 收 集 的 类 型 相符 。 


下 面 列 出 完整 的 解决 方案 。 设 定 为 注释 的 编写 《如 *1*) 便于 大 家 对 照 
程序 后 面 列 出 的 说 明 。 


//: RecycleB. java 

















// Adding more objects to the recycling problem 


package ci6.recycleb; 


import c16.trash.*; 
import java.util.*; 
// A vector that admits only the right type: 
class Tbin extends Vector { 
Class binType; 
Tbin(Class binType) { 
this.binType = binType; 
} 
boolean grab(Trash t) { 
// Comparing class types: 
if(t.getClass().equals(binType)) { 
addElement(t); 
return true; // Object grabbed 


} 


return false; // Object not grabbed 


} 
class TbinList extends Vector { //(*1*) 
boolean sort(Trash t) { 
Enumeration e = elements(); 
while(e.hasMoreElements()) { 
Tbin bin = (Tbhin)e.nextElement(); 


if(bin.grab(t)) return true; 


} 


return false; // bin not found for t 
} 
void sortBin(Tbin bin) { // (*2*) 
Enumeration e = bin.elements(); 
while(e.hasMoreElements()) 
if(!sort((Trash)e.nextElement())) 


System.out.println("Bin not found"); 


} 


public class RecycleB { 
static Tbin bin = new Tbin(Trash.class); 
public static void main(String[] args) { 
// Fill up the Trash bin: 
ParseTrash.fillBin("Trash.dat", bin); 
TbinList trashBins = new TbinList(); 
trashBins.addElement ( 
new Tbin(Aluminum.class) ); 
trashBins.addElement ( 
new Tbin(Paper.class)); 
trashBins.addElement ( 
new Tbin(Glass.class)); 


// add one line here: (*3*) 


trashBins.addElement ( 
new Tbin(Cardboard.class)); 

trashBins.sortBin(bin); // (*4*) 

Enumeration e = trashBins.elements(); 

while(e.hasMoreElements()) { 
Tbin b = (Tbhin)e.nextElement(); 
Trash.sumValue(b); 

} 

Trash.sumValue(bin); 

} 
} ///:~ 


(1) TbinList 容 纳 一 系列 Tbin 句 柄 ， 所 以 在 查找 与 我 们 传递 给 它 的 Trash 对 
象 相符 的 情况 时 ，sort0 能 通过 Tbin 继 承 。 


(2) sortBin0 人 允许 我 们 将 一 个 完整 的 Tbin 传 递 进 去 ， 而 且 它 会 在 Tbin 里 遍 
历 ， 挑 选 出 每 种 Trash， 并 将 其 归 类 到 特定 的 Tbin 中 。 请 注意 这 些 代 码 的 
通用 性 : 新 类 型 加 入 时 ， 它 本 喘 不 需要 任何 改动 。 只 要 新 类 型 加 入 (或 
发 生 其 他 事件 ) 时 大 量 代码 都 不 需要 变化 ， 就 表明 我 们 设计 的 是 一 个 容 
易 扩 展 的 系统 。 

(3) 现在 可 以 体会 添加 新 类 型 有 多 么 容易 了 。 为 文 持 添加 ， 只 需要 改动 
几 行 代 码 。 如 确实 有 必要 ， 甚 至 可 以 进一步 地 改进 设计 ， 使 更 多 的 代码 
都 保持 “固定 ”。 


(4) 一 个 方法 调用 使 bin 的 内 容 归 类 到 对 应 的 、 特 定 类 型 的 垃圾 简 里 。 











16.6 4 Ris 


上 述 设计 方案 肯定 是 令 人 满意 的 。 系 统 内 新 类 型 的 加 入 涉及 添加 或 修改 
不 同 的 类 ， 但 没有 必要 在 系统 内 对 代码 作 大 范围 的 改动 。 除 此 以 外 ， 
RTTI 并 不 象 它 在 RecycleA.java 里 那样 被 不 当地 使 用 。 然 而 ， 我 们 仍然 有 
可 能 更 深入 一 步 ， 以 最 “ 纯 ” 的 角度 来 看 竺 RTTI， 考 虑 如 何在 垃圾 分 类 系 
统 中 将 它 完全 消灭 。 


为 达到 这 个 目标 ， 首 先 必 须 认 识 到 : 对 所 有 与 不 同类 型 有 特殊 关联 的 活 
动 来 说 一 一 比如 侦 测 一 种 垃圾 的 具体 类 型 ， 并 把 它 置 入 适当 的 垃圾 人 简 里 
这 些 活动 都 应 当 通 过 多 形 性 以 及 动态 绑 定 加 以 控制 。 


以 前 的 例子 都 是 先 按 类 型 排序 ， 再 对 属于 茶 种 特殊 类 型 的 一 系列 元 素 进 
行 操 作 。 现 在 一 旦 需要 操作 特定 的 类 型 ， 吏 请 先 俘 下 来 想 一 想 。 事 实 
上 ， 多 形 性 《动态 绑 定 的 方法 调用 ) 整个 的 宗旨 就 是 帮 我 们 管理 与 不 同 
类 型 有 特殊 关联 的 信息 。 既 然 如 此 ， 为 什么 还 要 自己 去 检查 类 型 呢 ? 


答案 在 于 大 家 或 许 不 以 为 然 的 一 个 道理 ;Java 只 执行 单一 派 算 。 也 就 是 
说 ， 假 如 对 多 个 类 型 未 知 的 对 象 执 行 东 项 操作 ，Java 只 会 为 那些 类 型 中 
的 一 种 调用 动态 绑 定 机 制 。 这 当然 不 能 解决 问题 ， 所 以 最 后 不 得 不 人 工 
判断 荣 些 类 型 ， 才 能 有 效 地 产生 上 自己 的 动态 绑 定 行为 。 


为 解雇 这 个 缺 了 史 ， 我 们 需要 用 到 “多 重 派 站 ”机制 ， 这 意味 着 需要 建立 一 
个 配置 ， 使 单一 方法 调用 能 产生 多 个 动态 方法 调用 ， 从 而 在 一 次 处 理 过 
程 中 正确 判断 出 多 种 类 型 。 为 达到 这 个 要 求 ， 需 要 对 多 个 类 型 结构 进行 
操作 :每 一 次 派 遗 都 需要 一 个 类 型 结构 。 下 面 的 例子 将 对 两 个 结构 进行 
操作 : 现 有 的 Trash 系 列 以 及 由 垃圾 简 (Trash Bin) 的 类 型 构成 的 一 个 
系列 不 同 的 垃圾 或 废品 将 置 入 这 些 简 内 。 第 二 个 分 级 结构 并 非 绝 对 
显然 的 。 在 这 种 情况 下 ， 我 们 需要 人 为 地 创建 它 ， 以 执行 多 重 派 站 (H 
于 本 例 只 涉及 两 次 派 址 ， 所 以 称 为 “双重 派 址 >) 。 


16.6.1 实现 双重 派 遗 
记 住 多 形 性 只 能 通过 方法 调用 才能 表现 出 来 ， 所 以 假如 想 使 双重 派 遗 正 


确 进 行 ， 必 须 执行 两 个 方法 调用 : 在 每 种 结构 中 都 用 一 个 来 判断 其 中 的 
类 型 。 在 Trash 结 构 中 ， 将 使 用 一 个 新 的 方法 调用 addToBin()， 它 采用 的 



































参数 是 由 TypeBin 构 成 的 一 个 数组 。 那 个 方法 将 在 数组 中 遍历 ， 尝 试 将 
目 己 加 入 适当 的 垃圾 简 ， 这 里 正 是 双重 派遣 发 生 的 地 方 。 
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新 建立 的 分 级 结构 是 TypeBin， 其 中 包含 了 它 自己 的 一 个 方法 ， 名 为 
add0， 而 且 也 应 用 了 多 形 性 。 但 要 注意 一 个 新 特点 : add0 已 进行 了 “过 
载 ? 处 理 ， 可 接受 不 同 的 垃圾 类 型 作为 参数 。 因 此 ， 双 重 满足 机 制 的 一 
个 关键 点 是 它 也 要 涉及 到 过 载 。 

程序 的 重新 设计 也 带 来 了 一 个 问题 : 现在 的 基础 类 Trash 必 须 包 含 一 个 
addToBin(0) 方 法 。 为 解决 这 个 问题 ， 一 个 最 直接 的 办 法 是 复制 所 有 代 
人 码 ， 并 修改 基础 类 。 然 而 ， 假 如 没有 对 源码 的 控制 权 ， 那 么 还 有 男 一 个 
办 法 可 以 考虑 : 将 addToBin0) 方 法 置 入 一 个 接口 内 部 ， 保 持 Trash 不 变 ， 
并 继承 新 的 、 特 殊 的 类 型 Aluminum，Paper，Glass 以 及 Cardboard。 我 们 
在 这 里 准备 采取 后 一 个 办 法 。 

这 个 设计 方案 中 用 到 的 大 多 数 类 都 必须 设 为 public〈 公 用 ) 属性 ， 所 以 
它们 放置 于 自己 的 类 内 。 下 面 列 出 接口 代码 : 


//: TypedBinMember.java 


// An interface for adding the double dispatching 


// method to the trash hierarchy without 


// modifying the original hierarchy. 
package c16.doubledispatch; 
interface TypedBinMember { 

// The new method: 
boolean addToBin(TypedBin[] tb); 


Lif] fe 


在 Aluminum，Paper，Glass 以 及 Cardboard 每 个 特定 的 子 类 型 内 ， 都 会 实 
现 接口 TypeBinMember 的 addToBin() 方 法 ， 但 每 种 情况 下 使 用 的 代码 “ 似 
平 * 都 是 完全 一 样 的 : 


//: DDALuminum. java 


// Aluminum for double dispatching 
package ci6.doubledispatch; 
import ci6.trash.*; 
public class DDAluminum extends Aluminum 
implements TypedBinMember { 
public DDAluminum(double wt) { super(wt); } 
public boolean addToBin(TypedBin[] tb) { 
for(int i = 0; i < th.length; i++) 
if(tb[i].add(this) ) 
return true; 


return false; 


} ///:~ 


//: DDPaper.java 


// Paper for double dispatching 
package ci6.doubledispatch; 
import ci6.trash.*; 
public class DDPaper extends Paper 
implements TypedBinMember { 
public DDPaper (double wt) { super(wt); } 
public boolean addToBin(TypedBin[] tb) { 
for(int i = 0; i < tbh.length; i++) 
if(tb[i].add(this) ) 
return true; 
return false; 
} 
} ///:~ 


//: DDGlass.java 


// Glass for double dispatching 

package c16.doubledispatch; 

import ci6.trash.*; 

public class DDGlass extends Glass 
implements TypedBinMember { 


public DDGlass(double wt) { sSuper(wt); } 


public boolean addToBin(TypedBin[] tb) { 
for(int i = 0; i < th.length; i++) 
if(tb[i].add(this) ) 
return true; 
return false; 
} 
BAS A 


//: DDCardboard.java 


// Cardboard for double dispatching 
package ci6.doubledispatch; 
import c16.trash.*; 
public class DDCardboard extends Cardboard 
implements TypedBinMember { 
public DDCardboard(double wt) { super(wt); } 
public boolean addToBin(TypedBin[] tb) { 
for(int i = 0; i < th.length; i++) 
if(tb[i].add(this) ) 
return true; 


return false; 


MSL 


每 个 addToBin0 内 的 代码 会 为 数组 中 的 每 个 TypeBin 对 象 调用 add0。 但 
请 注意 参数 : this。 对 Trash 的 每 个 子 类 来 说 ，this 的 类 型 都 是 不 同 的 ， 所 
以 不 能 认为 代码 “完全 ”一 样 一 一 尽管 以 后 在 Java 里 加 入 参数 化 类 型 机 制 
后 便 可 认为 一 样 。 这 是 双重 派 遗 的 第 一 个 部 分 ， 因 为 一 旦 进入 这 个 方法 
内 部 ， 便 可 知道 到 底 是 Aluminum，Paper， 还 是 其 他 什么 垃圾 类 型 。 在 
对 add0 的 调用 过 程 中 ， 这 种 信息 是 通过 this 的 类 型 传递 的 。 编 译 器 会 分 
析出 对 add0 正 确 的 过 载 厂 本 的 调用 。 但 由 于 也 丘 会 产生 指向 基础 类 型 
TypeBin 的 一 个 句柄 ， 所 以 最 终 会 调用 一 个 不 同 的 方法 一 一 具体 什么 方 
法 取决 于 当前 选择 的 TypeBin 的 类 型 。 那 就 是 第 二 次 派 遗 。 


下 面 是 TypeBin 的 基础 类 : 


//: TypedBin. java 








// Nector that knows how to grab the right type 
package ci6.doubledispatch; 
import c16.trash.*; 
import java.util.*; 
public abstract class TypedBin { 
Vector v = new Vector(); 
protected boolean addIt(Trash t) { 
v.addElement(t); 
return true; 
} 
public Enumeration elements() { 
return v.elements(); 
} 


public boolean add(DDAluminum a) { 


return false; 


} 
public boolean add(DDPaper a) { 


return false; 


} 

public boolean add(DDGlass a) { 
return false; 

} 

public boolean add(DDCardboard a) { 


return false; 


} 
} ///:~ 


可 以 看 到 ， 过 载 的 add(0) 方 法 全 都 会 返回 false。 如 果 未 在 衍生 类 里 对 方法 
进行 过 载 ， 它 就 会 一 直人 返回 false， 而 且 调 用 者 (目前 是 addToBin()) 会 
认为 当前 Trash 对 象 尚 未 成 功 加 入 一 个 集合 ， 所 以 会 继续 查找 正确 的 集 


A 
A 


在 TypeBin 的 每 一 个 子 类 中 ， 都 只 有 一 个 过 载 的 方法 会 被 过 载 一 一 具体 
取决 于 准备 创建 的 是 什么 垃圾 位 类 型 。 举 个 例子 来 说 ，CardboardBin 会 
过 载 add(DDCardboard)。 过 载 的 方法 会 将 垃圾 对 象 加 入 它 的 集合 ， 并 返 
回 true。 而 CardboardBin 中 剩余 的 所 有 add0) 方 法 都 会 继续 返回 false， 
为 它们 尚未 过 载 。 事 实 上 ， 假 如 在 这 里 采用 了 参数 化 类 型 机 制 ，Java 代 
码 的 上 自动 创建 束 要 方便 得 多 《使 用 C++ 的 “模板 ?”， 我 们 不 必 费 事 地 为 子 
类 编码 ， 或 者 将 addToBin() 方 法 置 入 Trash 里 ，Java 在 这 方面 尚 有 待 改 
H) 


由 于 对 这 个 例子 来 说 ， 垃 圾 的 类 型 已 经 定制 并 置 入 一 个 不 同 的 目录 ， 所 
以 需要 用 一 个 不 同 的 垃圾 数据 文件 令 其 运转 起 来 。 下 面 是 一 个 示范 性 的 
DDTrash.dat: 
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DDGlass:54 


DDPaper :22 
DDPaper :11 
DDGlass:17 
DDALuminum: 89 
DDPaper : 88 
DDALuminum: 76 
DDCardboard: 96 
DDALuminum: 25 
DDALuminum: 34 
DDGlass:11 
DDGlass:68 
DDGlass:43 
DDALuminum: 27 
DDCardboard: 44 
DDALuminum: 18 
DDPaper :91 
DDGlass:63 
DDGlass:50 
DDGlass: 80 
DDALuminum: 81 


DDCardboard:12 


c16.DoubleDispatch.DDGlass:12 
c16.DoubleDispatch.DDGlass:54 
c16.DoubleDispatch.DDAluminum: 36 
c16.DoubleDispatch.DDAluminum: 93 
c16.DoubleDispatch.DDGlass: 93 
c16.DoubleDispatch.DDPaper : 80 
c16.DoubleDispatch.DDGlass: 36 
c16.DoubleDispatch.DDGlass:12 
c16.DoubleDispatch.DDGlass:60 
c16.DoubleDispatch.DDPaper : 66 
c16.DoubleDispatch.DDAluminum: 36 


c16.DoubleDispatch.DDCardboard: 22 


下 面 列 出 程序 剩余 的 部 分 : 


//: DoubleDispatch.java 


// Using multiple dispatching to handle more 
// than one unknown type during a method call. 
package c16.doubledispatch; 

import c16.trash.*; 

import java.util.*; 

class AluminumBin extends TypedBin { 


public boolean add(DDAluminum a) { 


return addIt(a); 


J 


class PaperBin extends TypedBin { 
public boolean add(DDPaper a) { 


return addIt(a); 


} 


class GlassBin extends TypedBin { 
public boolean add(DDGlass a) { 


return addIt(a); 


} 


class CardboardBin extends TypedBin { 
public boolean add(DDCardboard a) { 


return addIt(a); 


} 


class TrashBinSet { 
private TypedBin[] binSet = { 
new AluminumBin(), 
new PaperBin(), 


new GlassBin(), 


new CardboardBin( ) 
}; 
public void sortIntoBins(Vector bin) { 
Enumeration e = bin.elements(); 
while(e.hasMoreElements()) { 
TypedBinMember t = 
(TypedBinMember )e.nextElement(); 
if(!t.addToBin(binSet ) ) 


System.err.printin("Couldn't add " + t); 


} 
public TypedBin[] binSet() { return binSet; } 
} 
public class DoubleDispatch { 
public static void main(String[] args) { 
Vector bin = new Vector(); 
TrashBinSet bins = new TrashBinSet(); 
// ParseTrash still works, without changes: 
ParseTrash.fillBin("DDTrash.dat", bin); 
// Sort from the master bin into the 
// individually-typed bins: 
bins.sortIntoBins(bin); 


TypedBin[] tb = bins.binSet(); 


// Perform sumValue for each bin... 
for(int i = 0; i < tb.length; i++) 
Trash.sumValue(tb[i].v); 
// ... and for the master bin 
Trash.sumValue(bin); 
} 
} ///:~ 


其 中 ，TrashBinSet 封 装 了 各 种 不 同类 型 的 TypeBin， 同 时 还 有 
sortIntoBins() 方 法 。 所 有 双重 派 中 事件 都 会 在 那个 方法 里 发 生 。 可 以 看 
到 ， 一 旦 设置 好 结构 ， 再 归 类 成 各 种 TypeBin 的 工作 就 变 得 十 分 简单 
了 。 除 此 以 外 ， 两 个 动态 万 法 调用 的 效率 可 能 也 比 其 他 排序 方法 珊 一 








注意 这 个 系统 的 方便 性 主要 体现 在 main0 中 ， 同 时 还 要 注意 到 任何 特定 
的 类 型 信息 在 main0 中 都 是 完全 独立 的 。 只 与 Trash 基 础 类 接口 通信 的 其 
他 所 有 方法 都 不 会 受到 Trash 类 中 发 生 的 改变 的 干扰 。 


添加 新 类 型 需要 作出 的 改动 是 完全 孤立 的 : 我 们 随同 addToBin() 方 法 继 
承 Trash 的 新 类 型 ， 然 后 继承 一 个 新 的 TypeBin (这 实际 只 是 一 个 副本 ， 
可 以 简单 地 编辑 ) ， 最 后 将 一 种 新 类 型 加 入 TrashBinSet 的 集合 初 化 化 过 


程 。 


16.7 访问 髓 范式 


接 下 来 ， 让 我 们 思考 如 何 将 具有 完全 不 同 目标 的 一 个 设计 范式 应 用 到 世 
圾 归 类 系统 。 


对 这 个 范式 ， 我 们 不 再 关心 在 系统 中 加 入 新 型 Trash 时 的 优化 。 事 实 
上 ， 这 个 范式 使 新 型 Trash 的 添加 显得 更 加 复杂 。 假 定 我 们 有 一 个 基本 
类 结构 ， 它 是 固定 不 变 的 ， 它 或 许 来 自 男 一 个 开发 者 或 公司 ， 我 们 无 权 
对 那个 结构 进行 任何 修改 。 然 而 ， 我 们 又 希望 在 那个 结构 里 加 入 新 的 多 
形 性 方法 。 这 意味 着 我 们 一 般 必 须 在 基础 类 的 接口 里 添加 茶 些 东西 。 因 
此 ， 我 们 目前 面临 的 困境 是 一 方面 需要 问 基 础 类 添加 方法 ， 为 一 方面 义 
不 能 改动 基础 类 。 怎 样 解决 这 个 问题 呢 ? 


“访问 器 ”〈Visitor) 范式 使 我 们 能 扩展 基本 类 型 的 接口 ， 方 法 是 创建 类 
型 为 Visitor 的 一 个 独立 的 类 结构 ， 对 以 后 需 对 基本 类 型 采取 的 操作 进行 
虚拟 。 基 本 类 型 的 任务 束 是 简单 地 “接收 ”访问 器 ， 然 后 调用 访问 占 的 动 
态 绑 定 方法 。 看 起 来 就 象 下 面 这 样 : 
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accept(Visitor w) { 
vo visit(this); 














accept(Visitor w) { 
vo visit(this); 


accept(Visitor w) { 
vo visit(this); 


} } 


visit( Aluminum ) 
visit(Paper} 
visit( Glass) 


} 







visitt Aluminum) { visit Aluminum} { 
ff Perform Aluminum- ff Perform Aluminum =- 
ff spedfic work ¿i spedfic work 





visit Paper) { visit Paper) { 
ff Perform Paper- // Perform Paper- 
¿i spedfic work #/ spedfic work 


} } 

visit(Glass) { visit(Glass) { 
f/f Perform Glass- f/f Perform Glass- 
A spedfic work ¿i spedfic work 


} } 


现在 ， 假 如 vy 是 一 个 指向 Aluminum 〈 铝 制品 ) 的 Visitable 句 柄 ， 那 么 下 
述 代码 : 

PriceVisitor pv = new PriceVisitor(); 

v.accept(pv); 

会 造成 两 个 多 形 性 方法 调用 : 第 一 个 会 选择 accept0 的 Aluminum 版 本 ; 


第 二 个 则 在 acceptO 里 一 一 用 基础 类 Visitor 句 柄 v 动 态 调 用 visitO 的 特定 版 
本 时 。 





这 种 配置 意味 着 可 采取 Visitor 的 新 子 类 的 形式 将 新 的 功能 添加 到 系统 
里 ， 没 必要 接触 Trash 结 构 。 这 就 是 “访问 占 *” 范 式 最 主要 的 优点 : 可 为 一 
个 类 结构 添加 新 的 多 形 性 功能 ， 同 时 不 必 改 动 结构 只 要 安装 好 了 
accept() 方 法 。 注 意 这 个 优点 在 这 儿 是 有 用 的 ， 但 并 不 一 定 是 我 们 在 任 
何 情况 下 的 首选 方案 。 所 以 在 最 开始 的 时 候 ， 就 要 判断 这 到 搬 是 不 是 自 


己 需 要 的 方案 。 


现在 注意 一 件 没 有 做 成 的 事情 : 访问 器 方案 防止 了 从 主 控 Trash 序 列 向 
单独 类 型 序列 的 归 类 。 所 以 我 们 可 将 所 有 东西 都 留 在 单 主 探 序列 中 ， 只 
需 用 适当 的 访问 器 通过 那个 序列 传递 ， 即 可 达到 希望 的 目标 。 尽 管 这 似 
乎 并 非 访问 器 范式 的 本 意 ， 但 确实 让 我 们 达到 了 很 希望 达到 的 一 个 目标 
(避免 使 用 RTTI) 。 


访问 器 范式 中 的 双生 派 遗 负责 同时 判断 Trash 以 及 Visitor 的 类 型 。 在 下 面 
的 例子 中 ， 大 家 可 看 到 Visitor 的 两 种 实现 方式 : PriceVisitor 用 于 判断 总 
计 及 价格 ， 而 weightVisitor 用 于 跟踪 重量 。 


可 以 看 到 ， 所 有 这 些 都 是 用 回收 程序 一 个 新 的 、 改 进 过 的 版 本 实现 的 。 
而 且 和 DoubleDispatch.java 一 样 ，Trash 类 被 保持 孤立 ， 并 创建 一 个 新 接 
口 来 添加 accept(0) 方 法 : 














//: Nisitable.java 


// An interface to add visitor functionality to 
// the Trash hierarchy without modifying the 
// base class. 
package ci6.trashvisitor; 
import c16.trash.*; 
interface Visitable { 
// The new method: 
void accept(Visitor v); 


LIfe 


Aluminum，Paper，Glass 以 及 Cardboard 的 子 类 型 实现 了 accept(0) 方 法 : 


//: VALuminum. java 


// Aluminum for the visitor pattern 
package ci6.trashvisitor; 
import c16.trash.*; 
public class VAluminum extends Aluminum 
implements Visitable { 
public VAluminum(double wt) { Super(wt); } 
public void accept(Visitor v) { 
v.visit(this); 
} 
Lo PL 


//: \VPaper.java 


// Paper for the visitor pattern 
package ci6.trashvisitor; 
import ci6.trash.*; 
public class VPaper extends Paper 
implements Visitable { 
public VPaper(double wt) { Super(wt); } 


public void accept(Visitor v) { 


v.visit(this); 
} 
} ///:~ 


//: VGlass.java 


// Glass for the visitor pattern 
package ci6.trashvisitor; 
import ci6.trash.*; 
public class VGlass extends Glass 
implements Visitable { 
public VGlass(double wt) { super(wt); } 
public void accept(Visitor v) { 
v.visit(this); 
} 
于 


//: VCardboard. java 


// Cardboard for the visitor pattern 
package ci16.trashvisitor; 
import ci16.trash.*; 
public class VCardboard extends Cardboard 
implements Visitable { 
public VCardboard(double wt) { super(wt); } 


public void accept(Visitor v) { 


v.visit(this); 
} 
} ///:~ 





由 于 Visitor 基 础 类 没有 什么 需要 实在 的 东西 ， 可 将 其 创建 成 一 个 接口 : 


//: Visitor.java 


// The base interface for visitors 
package c16.trashvisitor, 
import ci16.trash.*; 
interface Visitor { 
void visit(VAluminum a); 
void visit(VPaper p); 
void visit(VGlass g); 
void visit(VCardboard c); 
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c16.TrashVisitor.VGlass:12 
c16.TrashVisitor.VGlass:60 
c16.TrashVisitor.VPaper :66 
c16.TrashVisitor.VALuminum: 36 


ci6.TrashVisitor.VCardboard: 22 


程序 剩余 的 部 分 将 创建 特定 的 Visitor 类 型 ， 并 通过 一 个 Trash 对 象 列表 发 
送 它们 : 


//: TrashVisitor.java 


// The "visitor" pattern 

package ci6.trashvisitor; 

import c16.trash.*; 

import java.util.*; 

// Specific group of algorithms packaged 

// in each implementation of Visitor: 

class PriceVisitor implements Visitor { 

private double alSum; // Aluminum 

private double pSum; // Paper 

private double gSum; // Glass 

private double cSum; // Cardboard 

public void visit(VAluminum al) { 
double v = al.weight() * al.value(); 


System.out.printin( 


"value of Aluminum= " + v); 
alSum += v; 
} 
public void visit(VPaper p) { 
double v = p.weight() * p.value(); 
System.out.printin( 
"value of Paper= " + v); 
pSum += v; 
} 
public void visit(VGlass g) { 
double v = g.weight() * g.value(); 
System.out.printin( 
"value of Glass= " + v); 
gSum += V; 
} 
public void visit(VCardboard c) { 
double v = c.weight() * c.value(); 
System.out.printin( 
"value of Cardboard = " + v); 
cSum += vV; 
} 
void total() { 


System,.out.printJln( 


"Total Aluminum: $" + alSum + "\n" + 
"Total Paper: $" + pSum + "\n" + 
"Total Glass: $" + gSum + "\n" + 


"Total Cardboard: $" + cSum); 


} 
class WeightVisitor implements Visitor { 
private double alSum; // Aluminum 
private double pSum; // Paper 
private double gSum; // Glass 
private double cSum; // Cardboard 
public void visit(VAluminum al) { 
alSum += al.weight(); 
System.out.printin("weight of Aluminum = " 
+ al.weight()); 
} 
public void visit(VPaper p) { 
pSum += p.weight(); 
System.out.printin("weight of Paper = " 
+ p.weight()); 
} 
public void visit(VGlass g) { 


gSum += g.weight(); 


System.out.printin("weight of Glass = " 
+ g.weight()); 
} 
public void visit(VCardboard c) { 
cSum += c.weight(); 
System.out.println("weight of Cardboard = " 
+ c.weight()); 
} 
void total() { 
System.out.println("Total weight Aluminum:" 
+ alSum); 
System.out.println("Total weight Paper:" 
+ pSum) ; 
System.out.printin( "Total weight Glass:" 
+ gSum); 
System.out.printin("Total weight Cardboard:" 


+ cSum); 


} 


public class TrashVisitor { 
public static void main(String[] args) { 
Vector bin = new Vector(); 


// ParseTrash still works, without changes: 


ParseTrash.fillBin("VTrash.dat", bin); 
// You could even iterate through 
// a list of visitors! 
PriceVisitor pv = new PriceVisitor(); 
WeightVisitor wv = new WeightVisitor(); 
Enumeration it = bin.elements(); 
while(it.hasMoreElements()) { 
Visitable v = (Visitable)it.nextElement(); 
v.accept(pv); 
v.accept(wv); 
} 
pv.total(); 
wv.total(); 
} 
} ///:~ 


注意 main() 的 形状 已 再 次 发 生 了 变化 。 现 在 只 有 一 个 垃圾 (Trash) fo 
两 个 Visitor 对 象 被 接收 到 序列 中 的 每 个 元 素 内 ， 它 们 会 完成 自己 份 内 的 
工作 。Visitor 跟 踪 它 们 自己 的 内 部 数据 ， 计 算出 总 重 和 价格 。 


最 好 ， 将 东西 从 序列 中 取出 的 时 候 ， 除 了 不 可 避免 地 癌 Trash 造 型 以 
外 ， 和 再 没有 运行 期 的 类 型 验证 。 知 在 Java 里 实现 了 参数 化 类 型 ， 甚 全 那 
个 造型 操作 也 可 以 避免。 


对 比 之 前 介绍 过 的 双重 派 遗 方案 ， 区 分 这 两 种 方案 的 一 个 办 法 是 : 在 双 
重 派 中 方案 中 ， 每 个 子 类 创建 时 只 会 过 载 其 中 的 一 个 过 载 方法 ， 即 
add0。 而 在 这 里 ， 每 个 过 载 的 visit0) 方 法 都 必须 在 Visitor 的 每 个 子 类 中 
进行 过 载 。 








1. 更 多 的 结合 ? 


这 里 还 有 其 他 许多 代码 ，Trash 结 构 和 Visitor 结 构 之 间 存 在 着 明显 的 “ 结 
A” (Coupling) 关系 。 然 而 ， 在 它们 所 代表 的 类 集 内 部 ， 也 存在 着 高 度 
的 凝聚 力 : 都 只 做 一 件 事情 〈Trash 描 述 垃圾 或 废品 ， 而 Visitor 描 述 对 垃 
圾 采取 什么 行动 ) 。 作 为 一 套 优秀 的 设计 方案 ， 这 无 疑 是 个 良好 的 开 
端 。 当 然 就 目前 的 情况 来 说 ， 只 有 在 我 们 添加 新 的 Visitor 类 型 时 才能 体 
会 到 它 的 好 处 。 但 在 添加 新 类 型 的 Trash 时 ， 它 却 显得 有 些 碍 手 碍 脚 。 


类 与 类 之 间 低 度 的 结合 与 类 内 高 度 的 凝聚 无 疑 是 一 个 重要 的 设计 目标 。 
但 只 要 稍 不 留神 ， 就 可 能 妨碍 我 们 得 到 一 个 本 该 更 出 色 的 设计 。 从 表面 
看 ， 有 些 类 不 可 避免 地 相互 间 存 在 着 一 些 “ 杀 密 ” 关 系 。 这 种 关系 通常 是 
成 对 发 生 的 ， 可 以 叫 作 “ 对 联 ”(Couplet) 比如 集合 和 继承 右 
(Enumeration) 。 前 面 的 Trash-Visitor 对 似乎 也 是 这 样 的 一 种 “对 联 ”。 














16.8 RTTI 真 的 有 害 吗 


本 章 的 各 种 设计 方案 都 在 努力 避免 使 用 RTTI， 这 或 许 会 给 大 家 留 

下 “RTTI 有 害 ” 的 印象 (还 记得 可 怜 的 goto 吗 ， 由 于 给 入 印象 不 佳 ， 根 本 
就 没有 放 到 Java 里 来 ) 。 但 实际 情况 并 非 绝 对 如 此 。 正 确 地 次 ， 应 该 是 
RTTI 使 用 不 当 才 “有 害 ”。 我 们 之 所 以 想 避 免 RTTI 的 使 用 ， 是 由 于 它 的 
错误 运用 会 造成 扩展 性 受到 损害 。 而 我 们 事前 提出 的 目标 就 是 能 向 系统 
目 由 加 入 新 类 型 ， 同 时 保证 对 周围 的 代码 造成 尽 可 能 小 的 影响 。 由 于 

RTTI 常 被 滥用 (让 它 伍 找 系统 中 的 每 一 种 类 型 )， 会 造成 代码 的 扩展 
能 力 大 打折 扣 一 一 添加 一 种 新 类 型 时 ， 必 须 找 出 使 用 了 RTTI 的 所 有 代 
码 。 即 使 仅 遗 漏 了 其 中 的 一 个 ， 也 不 能 从 编译 器 那里 得 到 任何 帮助 。 


然而 ，RTTI 本 里 并 不 会 自动 产生 非 扩 展 性 的 代码 。 让 我 们 再 来 看 一 看 
前 面 提 到 的 垃圾 回收 例子 。 这 一 次 准备 引入 一 种 新 工具 ， 我 把 它 叫 作 
TypeMap。 其 中 包含 了 一 个 Hashtable〈 散 列表 ) ， 其 中 容纳 了 多 个 
Vector， 但 接口 非常 简单 : 可 以 添加 CaddQ) 一 个 新 对 象 ， 可 以 获得 
(get()) 一 个 Vector， 其 中 包含 了 属于 某 种 特定 类 型 的 所 有 对 象 。 对 于 
这 个 包含 的 散 列 表 ， 它 的 关键 在 于 对 应 的 Vector 里 的 类 型 。 这 种 设计 方 
案 的 优点 (根据 Lary ”O'Brien 的 建议 ) 是 在 过 到 一 种 新 类 型 的 时 候 ， 
TypeMap 会 动态 加 入 一 种 新 类 型 。 所 以 不 管 什 么 时 候 ， 只 要 将 一 种 新 类 
型 加 入 系统 《即使 在 运行 期 间 添 加 ) ， 它 也 会 正确 无 误 地 得 以 接受 。 


我 们 的 例子 同样 建立 在 c16.Trash 这 个 “ 包 ”(Package〉 内 的 Trash 类 型 结 
构 的 基础 上 《而 且 那 儿 使 用 的 Trash.dat 文 件 可 以 照搬 到 这 里 来 ) 。 


//: DynaTrash.java 








// Using a Hashtable of Vectors and RTTI 
// to automatically sort trash into 

// vectors. This solution, despite the 
// use of RTTI, is extensible. 


package ci6.dynatrash; 


import c16.trash.*; 
import java.util.*; 
// Generic TypeMap works in any situation: 
class TypeMap { 
private Hashtable t = new Hashtable(); 
public void add(Object o) { 
Class type = o.getClass(); 
if(t.containsKey(type) ) 
((Vector)t.get(type) ).addElement(o); 
else { 
Vector v = new Vector(); 


v.addElement(o); 


t.put(type,v); 


i 


public Vector get(Class type) { 
return (Vector)t.get(type); 
} 
public Enumeration keys() { return t.keys(); } 
// Returns handle to adapter class to allow 
// callbacks from ParseTrash.fillBin(): 
public Fillable filler() { 


// Anonymous inner class: 


return new Fillable() { 
public void addTrash(Trash t) { add(t); } 


}; 


} 
public class DynaTrash { 
public static void main(String[] args) { 
TypeMap bin = new TypeMap(); 
ParseTrash.fillBin("Trash.dat",bin.filler()); 
Enumeration keys = bin.keys(); 
while(keys.hasMoreElements() ) 
Trash. sumValue( 
bin.get((Class)keys.nextElement())); 
} 
} ///:~ 





尽管 功能 很 强 ， 但 对 TypeMap 的 定义 是 非常 简单 的 。 它 只 是 包含 了 一 个 
散 列 表 ， 同 时 add0 负 担 了 大 部 分 的 工作 。 添 加 一 个 新 类 型 时 ， 那 种 类 型 
的 Class 对 象 的 句柄 会 被 提取 出 来 。 随 后 ， 利 用 这 个 句柄 判断 容纳 了 那 燃 
对 象 的 一 个 Vector 是 否 已 存在 于 散 列 表 中 。 如 答案 是 肯定 的 ， 融 提取 出 
那个 Vector， 并 将 对 象 加 入 其 中 ; 反之， 就 将 Class 对 象 及 新 Vector 作为 
m a eT 


利用 keysO0， 可 以 得 到 对 所 有 Class 对 象 的 一 个 “ 枚 举 ”(Enumeration , 
而 且 可 用 get0， 可 通过 Class 对 象 获 取 对 应 的 Vector。 


filler() 方 法 非常 有 趣 ， 因 为 它 利 用 了 ParseTrash.fillBin() 的 设计 一 一 不 仅 
能 尝试 填充 一 个 Vector， 也 能 用 它 的 addTrash0) 方 法 试 着 填充 实现 了 




















Fillable (可 填充 ) 接口 的 任何 东西 。filter() 需 要 做 的 全 部 事情 就 是 将 一 
个 句柄 返回 给 实现 了 Fillable 的 一 个 接口 ， 然 后 将 这 个 句柄 作为 参数 传递 
给 filBin0， 束 象 下 面 这 样 


ParseTrash.fillBin(""Trash.dat", bin.filler()); 


为 产生 这 个 句柄 ， 我 们 采用 了 一 个 “匿名 内 部 类 ”《〈 已 在 第 7 章 讲述 ) 。 
由 于 根本 不 需要 用 一 个 已 命名 的 类 来 实现 Fillable， 只 需要 属于 那个 类 的 
一 个 对 象 的 句柄 即 可 ， 所 以 这 里 使 用 匿名 内 部 类 是 非常 恰当 的 。 


对 这 个 设计 ， 要 注意 的 一 个 地 方 是 尽管 没有 设计 成 对 归 类 加 以 控制 ， 但 
在 fllBin0 每 次 进行 归 类 的 时 候 ， 都 会 将 一 个 Trash 对 象 插入 bin。 


通过 前 面 那 些 例 子 的 学 习 ，DynaTrash 类 的 大 多 数 部 分 都 应 当 非 常熟 悉 
了 。 这 一 次 ， 我 们 不 再 将 新 的 Trash 对 象 置 入 类 型 Vector 的 一 个 bin 内 。 
由 于 bin 的 类 型 为 TypeMap， 所 以 将 垃圾 (Trash) BPEL (Bin) 的 
时 候 ，TypeMap 的 内 部 归 类 机 制 会 立即 进行 适当 的 分 类 。 在 TypeMap 里 
通 历 并 对 每 个 独立 的 Vector 进行 操作 ， 这 是 一 件 相 当 人 简单 的 事情 : 


Enumeration keys = bin.keys(); 











while(keys.hasMoreElements() ) 
Trash. sumValue ( 


bin.get((Class)keys.nextElement())); 


就 象 大 家 看 到 的 那样 ， 新 类 型 向 系统 的 加 入 根本 不 会 影响 到 这 些 代 码 ， 
亦 不 会 影响 TypeMap 中 的 代码 。 这 显然 是 解雇 问题 最 圆满 的 方案 。 尽 管 
它 确 实 严 重 依赖 RTTI， 但 请 注意 散 列 表 中 的 每 个 键 一 值 对 都 只 查找 一 
种 类 型 。 除 此 以 外 ， 在 我 们 增加 一 种 新 类 型 的 时 候 ， 不 会 陷入 “ 怎 记 ” 问 
系统 加 入 正确 代码 的 尴 碎 境地 ， 因 为 根本 就 没有 什么 代码 需要 添加 。 


16.9 总 结 


从 表面 看 ， 由 于 象 TrashVisitor.java 这 样 的 设计 包含 了 比 早 期 设计 数量 更 
多 的 代码 ， 所 以 会 留 下 效率 不 高 的 印象 。 试 图 用 各 种 设计 方案 达到 什么 
目的 应 该 是 我 们 考虑 的 重点 。 设 计 范 式 特别 适合 “将 发 生变 化 的 东西 与 
保持 不 变 的 东西 隔离 开 ”。 而 “发 生变 化 的 东西 ”可 以 代表 许多 种 变化 。 
之 所 以 发 生变 化 ， 可 能 是 由 于 程序 进入 一 个 新 环境 ， 或 者 由 于 当前 环境 
的 一 些 东西 发 生 了 变化 《〈“ 例 如“ 用户 和 希望 在 屏幕 上 当前 显示 的 网 示 中 添 
加 一 种 新 的 几何 形状 >) BAAS Fe FIA A SE, AREY EAE MT AC 
BERRIDGE. SN Ri PRA A AY VA TAY Trash lA] AK 
的 加 入 ， 但 TrashVisitor.java 人 允许 我 们 方便 地 添加 新 功能 ， 同 时 不 会 对 
Trash 结 构造 成 干扰 。TrashVisitor.java 里 确实 多 出 了 许多 代码 ， 但 在 
Visitor 里 添加 新 功能 只 需要 极 小 的 代价 。 如 果 经 常 都 要 进行 此 类 活动 ， 
那么 多 一 些 代 码 也 是 值得 的 。 


变化 序列 的 发 现 并 非 一 件 平常 事 ， 在 程序 的 初始 设计 出 台 以 前 ， 那 些 分 
析 家 一 般 不 可 能 预测 到 这 种 变化 。 除 非 进入 项 目 设计 的 后 期 ， 人 否则 一 些 
必要 的 信息 是 不 会 显露 出 来 的 ;， 有 时 只 有 进入 设计 或 最 终 实现 阶段 ， 才 
能 体会 到 对 自己 系统 一 个 更 深入 或 更 不 易 察觉 需要 。 添 加 新 类 型 时 (这 
是 “回收 ”例子 最 主要 的 一 个 重点 ) ， 可 能 会 意识 到 只 有 自己 进入 维护 阶 
段 ， 而 且 开始 扩 充 系 统 时 ， 才 需要 一 个 特定 的 继承 结构 。 


通过 设计 范式 的 学 习 ， 大 家 可 体会 到 最 重要 的 一 件 事情 就 是 本 书 一 直 宣 
扬 的 一 个 观点 一 一 多 形 性 是 OOP 《面向 对 象 程序 设计 ) 的 全 部 一 一 已 发 
生 了 彻底 的 改变 。 换 句 话 说 ， 很 难 “ 获 得 ”多 形 性 ， 而 一 旦 获得 ， 就 需要 
尝试 将 自己 的 所 有 设计 都 造型 到 一 个 特定 的 模子 里 去 。 


设计 范式 要 表明 的 观点 是 “OOP 并 不 仅仅 同 多 形 性 有 关 ”。 应 当 与 OOP 有 
天 的 是 “将 发 生变 化 的 东西 同 保持 不 变 的 东西 分 隔 开 来 ”。 多 形 性 是 达到 
这 一 目的 的 特别 重要 的 手段 。 而 且 假 如 编程 语言 直接 文 持 多 形 性 ， 那 么 
它 束 显得 尤其 有 用 《由 于 直接 文 持 ， 所 以 不 必 目 己 动 手 编写 ， 从 而 节省 
大 量 的 精力 和 时 间 ) 。 但 设计 范式 同 我 们 揭示 的 却 是 达到 基本 目标 的 忆 
EER. MA BASH ee SEAS, ate ROA OY Mi 
出 更 有 创新 性 的 设计 。 


由 于 《Design Patterns》 这 本 书 对 程序 员 造 成 了 如 此 重要 的 影响 ， 所 以 
























































他 们 纷纷 开始 寻找 其 他 范式 。 随 着 的 时 间 的 推移 ， 这 类 范式 必然 会 越 来 
越 多 。JimCoplien (http:/www.bell-labs.com/~cope 主 页 作者 ) 向 我 们 推 
荐 了 这 样 的 一 些 站 点 ， 上 面 有 许多 很 有 价值 的 范式 说 明 : 
http://st-www.cs.uiuc.edu/users/patterns 

http://c2.com/cgi/wiki 

http://c2.com/ppr 
http://www.bell-labs.com/people/cope/Patterns/Process/index.html 
http://www. bell-labs.com/cgi-user/OrgPatterns/OrgPatterns 
http://st-www.cs.uiuc.edu/cgi-bin/wikic/wikic 


http://www.cs.wustl.edu/~schmidt/patterns.html 


http://www.espinc.com/patterns/overview. html 


同时 请 留意 每 年 都 要 召开 一 届 权 威 性 的 设计 范式 会 议 ， 名 为 PLOP。 会 
议会 出 版 许多 学 术 论 文 ， 第 三 届 已 在 1997 年 底 召 开 过 了 ， 会 议 所 有 资料 
均 由 Addison-Wesley 出 版 。 


16.10 练习 


(1) 将 SingletonPattern.java 作 为 起 点 ， 创 建 一 个 类 ， 用 它 管理 目 己 固定 数 
量 的 对 象 。 


(2) 为 TrashVisitor.java 添 加 一 个 名 为 Plastic〈 塑 料 ) 的 类 。 


(3) 为 DynaTrash.java 同 样 添 加 一 个 Plastic《〈 塑 料 ) 类 。 
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本 章 包 含 了 一 系列 项 目 ， 它 们 都 以 本 书 介绍 的 内 容 为 基础 ， 并 对 早期 的 
革 市 进行 了 一 定 程度 的 扩充 。 


与 以 前 经 历 过 的 项 目 相 比 ， 这 儿 的 大 多 数 项 目 都 明显 要 复杂 得 多 ， 它 们 
充分 演示 了 新 技术 以 及 类 库 的 运用 。 





17.1 文字 处 理 


如 果 您 有 C 或 C++ 的 经 验 ， 那 么 最 开始 可 能 会 对 Java 控 制 文本 的 能 力 感 
到 怀疑 。 事实 上 ， 我 们 最 害怕 的 残 是 速度 特别 慢 ， 这 可 能 妨碍 我 们 创造 
能 力 的 发 挥 。 然 而 ，Java 对 应 的 工具 《特别 是 String 类 ) 具有 很 强 的 功 
能 ， 就 象 本 节 的 例子 展示 的 那样 〈 而 且 性 能 也 有 一 定 程度 的 提升 ) 。 


正如 大 家 即将 看 到 的 那样 ， 建 立 这 些 例子 的 目的 都 是 为 了 解决 本 书 编制 
过 程 中 遇 到 的 一 些 问 题 。 但 是 ， 它 们 的 能 力 并 非 仅 止 于 此 。 通 过 人 简单 的 
改造 ， 即 可 让 它们 在 其 他 场合 大 显 喘 手 。 除 此 以 外 ， 它 们 还 揭示 出 了 本 
书 以 前 没有 强调 过 的 一 项 Java 特 性 。 


17.1.1 提取 代码 列表 


对 于 本 书 每 一 个 完整 的 代码 列表 (不 是 代码 段 ，， 大 家 无 疑 会 注意 到 它 
们 都 用 特殊 的 注释 记号 起 始 与 结束 CW:' 和 WW/:~') 。 之 所 以 要 包括 这 种 
标志 信息 ， 是 为 了 能 将 代码 从 本 书 自 动 提取 到 兼容 的 源码 文件 中 。 在 我 
的 前 一 本 书 里 ， 我 设计 了 一 个 系统 ， 可 将 测试 过 的 代码 文件 自动 合并 到 
书 中 。 但 对 于 这 本 书 ， 我 发 现 一 种 更 简便 的 做 法 是 一 旦 通过 了 最 初 的 测 
试 ， 就 把 代码 粘贴 到 书 中 。 而 且 由 于 很 难 第 一 次 就 编译 通过 ， 所 以 我 在 
书 的 内 部 编辑 代码 。 但 如 何 提取 并 测试 代码 昵 ? 这 个 程序 就 是 关键 。 如 
果 你 打算 解决 一 个 文字 处 理 的 问题 ， 那 么 它 也 很 有 利用 价值 。 该 例 也 演 
示 了 String 类 的 许多 特性 。 


我 首先 将 整 本 书 都 以 ASCII 文 本 格式 保存 成 一 个 独立 的 文件 。 
CodePackager 程 序 有 两 种 运行 模式 (在 usageString 有 相应 的 描述 ) : 如 
果 使 用 -p 标 志 ， 程 序 就 会 检查 一 个 包含 了 ASCII 文 本 〈 即 本 书 的 内 容 ) 
的 一 个 输入 文件 。 它 会 表 历 这 个 文件 ， 按 照 注释 记号 提取 出 代码 ， 并 用 
位 于 第 一 行 的 文件 名 来 决定 创建 文件 使 用 什么 名 字 。 除 此 以 外 ， 在 需要 
将 文件 置 入 一 个 特殊 目录 的 时 候 ， 它 还 会 检查 package 语 句 〈( 根 据 由 
package 语 句 指定 的 路 径 选 择 ) 。 


但 这 样 还 不 够 。 程 序 还 要 对 包 (package) 名 进行 跟踪 ， 从 而 监视 章 内 
发 生 的 变化 。 由 于 每 一 草 使 用 的 所 有 包 都 以 c02，c03，c04 等 等 起 头 ， 

用 于 标记 它们 所 属 的 是 哪 一 革除 那些 以 com 起 头 的 以 外 ， 它 们 在 对 不 
同 的 章 进 行 跟 踩 的 时 候 会 被 忽略 ) 只 要 每 一 章 的 第 一 个 代码 列表 包 

















含 了 一 个 package， 所 以 CodePackager 程 序 能 知道 每 一 章 发 生 的 变化 ， 并 
将 后 名 卖 的 文件 放 到 新 的 子 目录 里 。 


每 个 文件 提取 出 来 时 ， ee ,随后 再 将 那 

个 对 象 置 入 一 个 集合 (后 面 还 会 详尽 讲述 这 个 过 程 )。 

SourceCodeFile 对 象 可 以 简单 地 保存 在 文件 中 ， 人 的 第 二 

用 途 。 如 果 直 接 调用 CodePackager， 不 添加 -p 标 志 ， 它 就 会 将 一 个 ' 人 

包 ” 文 件 作 为 输入 。 那 个 文件 随后 会 被 提取 《释放 ) 进入 单独 的 文件 。 

a eee cane (packed) 进入 这 个 
— rf 文 > 


但 为 什么 还 要 如 此 麻烦 地 使 用 打包 文件 呢 ? 这 是 由 于 不 同 的 计算 机 平台 
用 不 同 的 方式 在 文件 里 保存 文本 信息 。 其 中 最 大 的 问题 是 换行 字符 的 表 
未 方法 ; 当然 ， 还 有 可 能 存在 为 一 些 问题 。 然 而 ，Java 有 一 种 特殊 类 型 
的 IO 数据 流 DataOutputStream 它 可 以 保证 “无 论 数据 来 自 何 种 机 
器 ， 只 要 使 用 一 个 DataInputStream 收 取 这 些 数据 ， 就 可 用 本 机 正确 的 格 
式 保存 它们 ”。 也 就 是 说 ，Java 负 责 控制 与 不 同 平台 有 关 的 所 有 细 市 ， 

而 这 正 是 Java 最 具 魅 力 的 一 点 。 所 以 -p 标 志 能 将 所 有 东西 都 保存 到 单一 
SCE E; 并 采用 通用 的 格式 。 用 户 可 从 Web 下 载 这 个 文件 以 及 Java 程 
序 ， 然 后 对 这 个 文件 运行 CodePackager， 同时 个 指定 -p 标 记 ， 文件 便 会 
释放 到 系统 中 正确 的 场所 〈 亦 可 指定 另 一 个 子 目 录 ; 否则 就 在 当前 目录 
BEF AS) 。 为 确保 不 会 留 下 与 特定 平台 有 关 的 格式 ， 几 是 需要 摘 述 
一 个 文件 或 路 径 的 时 候 ， 我 们 就 使 用 File 对 象 。 除 此 以 外 ， 还 有 一 项 特 
别 的 安全 措施 : 在 每 个 子 目 录 里 都 放 入 一 个 空 文件 ; 那个 文件 的 名 字 指 
出 在 那个 子 目 录 里 应 找到 多 少 个 文件 。 


下 面 是 完整 的 代码 ， 后 面 会 对 它 进行 详细 的 说 明 : 


//: CodePackager.java 












































// "Packs" and "unpacks" the code in "Thinking 

// in Java" for cross-platform distribution. 

/* Commented so CodePackager sees it and starts 
a new chapter directory, but so you don't 


have to worry about the directory where this 


program lives: 
package c17; 
*/ 
import java.util.*; 
import java.io.*; 
class Pr { 
static void error(String e) { 
System.err.println( "ERROR: ”+ e); 


System.exit(1); 


} 


class IO { 
static BufferedReader disOpen(File f) { 
BufferedReader in = null; 
try { 
in = new BufferedReader ( 
new FileReader(f)); 
} catch(IOException e) { 
Pr.error("could not open " + f); 
} 


return in; 


} 


static BufferedReader disOpen(String fname) 


return disOpen(new File(fname) ); 
} 
static DataOutputStream dosOpen(File f) { 
DataOutputStream in = null; 
try { 
in = new DataOutputStream( 
new BufferedOutputStream( 
new FileOutputStream(f))); 
} catch(IOException e) { 
Pr.error("could not open " + f); 


} 


return in; 
} 
static DataOutputStream dosOpen(String fname) { 
return dosOpen(new File(fname) ); 
} 
static PrintWriter psOpen(File f) { 
PrintWriter in = null; 
try { 
in = new PrintWriter( 
new Bufferedwriter ( 
new Filewriter(f))); 


} catch(I0OException e) { 


Pr.error("could not open " + f); 
} 
return in; 
} 
static Printwriter psOpen(String fname) { 
return psOpen(new File(fname)); 
上 
static void close(Writer os) { 
try { 
os.close(); 
} catch(IOException e) { 


Pr.error("closing " + os); 


} 


static void close(DataOutputStream os) { 


try { 


os.close(); 
} catch(IOException e) { 


Pr.error("closing " + os); 


} 


static void close(Reader os) { 


try { 


os.close(); 
} catch(IOException e) { 


Pr.error("closing " + os); 


} 


class SourceCodeFile { 
public static final String 
startMarker = "//:", // Start of source file 
endMarker = "} ///:~", // End of source 
endMarker2 = "}; ///:~", // C++ file end 
beginContinue = "} ///:Continued", 
endContinue = "///:Continuing", 
packMarker = "###", // Packed file header tag 
eol = // Line separator on current system 
System.getProperty("line.separator"), 
filesep = // System's file path separator 
System.getProperty("file.separator"); 
public static String copyright = ""; 
static { 
try { 
BufferedReader cr = 


new BufferedReader ( 


new FileReader("Copyright.txt")); 
String crin; 
while((crin = cr.readLine()) != null) 
copyright += crin + "\n"; 
cr.close(); 
} catch(Exception e) { 


copyright = ""; 


} 
private String filename, dirname, 
contents = new String(); 
private static String chapter = "c02"; 
// The file name separator from the old system: 
public static String oldsep; 
public String toString() { 
return dirname + filesep + filename; 
} 
// Constructor for parsing from document file: 
public SourceCodeFile(String firstLine, 
BufferedReader in) { 
dirname = chapter; 
// Skip past marker: 


filename = firstLine.substring( 


startMarker.length()).trim(); 
// Find space that terminates file name: 
if(filename.indexOf(' ') != -1) 
filename = filename. substring( 
©, filename.indexOf(' ')); 
System.out.printin("found: " + filename); 
contents = firstLine + eol; 
if(copyright.length() != 0) 
contents += copyright + eol; 
String s; 
boolean foundEndMarker = false; 
try { 
while((s = in.readLine()) != null) { 
if(s.startswith(startMarker ) ) 
Pr.error("No end of file marker for " + 
filename); 
// For this program, no spaces before 
// the "package" keyword are allowed 
// in the input source code: 
else if(s.startswith("package")) { 
// Extract package name: 
String pdir = s.substring( 


s.indexOf(' ')).trim(); 


pdir = pdir.substring( 

©, pdir.indexOf(';')).trim(); 

// Capture the chapter from the package 
// ignoring the 'com' subdirectories: 
if(!pdir.startswith("com")) { 

int firstDot = pdir.indexOf('.'); 

if(firstDot != -1) 

chapter = 
pdir.substring(0,firstDot); 
else 
chapter = pdir; 
} 
// Convert package name to path name: 
pdir = pdir.replace( 

'.', filesep.charAt(0)); 
System.out.println("package " + pdir); 
dirname = pdir; 

} 
contents += s + eol; 
// Move past continuations: 
if(s.startswith(beginContinue) ) 
while((s = in.readLine()) != null) 


if(s.startswith(endContinue)) { 


contents += S + eol; 
break; 
} 
// Watch for end of code listing: 
if(s.startswith(endMarker) | | 
s.startswith(endMarker2)) { 
foundEndMarker = true; 


break; 


} 
if(!foundEndMarker) 
Pr.error( 
"End marker not found before EOF"); 
System.out.printin("Chapter: " + chapter); 
} catch(IOException e) { 


Pr.error("Error reading line"); 


h; 


// For recovering from a packed file: 
public SourceCodeFile(BufferedReader pFile) { 
try { 
String s = pFile.readLine(); 


if(s == null) return; 


if(!s.startswith(packMarker ) ) 
Pr.error("Can't find " + packMarker 
saa i o SS 
s = s.substring( 
packMarker.length()).trim(); 
dirname = s.substring(0, s.indexOf('#')); 
filename = s.substring(s.indexOf('#') + 1); 
dirname = dirname.replace( 
oldsep.charAt(0), filesep.charAt(@)); 
filename = filename.replace( 
oldsep.charAt(0), filesep.charAt(0)); 
System.out.printin("listing: " + dirname 
+ filesep + filename); 
while((s = pFile.readLine()) != null) { 
// Watch for end of code listing: 
if(s.startswith(endMarker) || 
s.startswith(endMarker2)) { 
contents += s; 
break; 


} 


contents += s + eol; 


} 


} catch(IOException e) { 


System.err.printin("Error reading line"); 


} 


public boolean hasFile() { 
return filename != null; 
} 
public String directory() { return dirname; } 
public String filename() { return filename; } 
public String contents() { return contents; } 
// To write to a packed file: 
public void writePacked(DataOutputStream out) { 
try { 
out .writeBytes( 
packMarker + dirname + "#" 
+ filename + eol); 
out.writeBytes(contents); 
} catch(IOException e) { 
Pr.error("writing " + dirname + 


filesep + filename); 


} 


// To generate the actual file: 


public void writeFile(String rootpath) { 


File path = new File(rootpath, dirname) ; 
path.mkdirs(); 
PrintWriter p = 

I0.psOpen(new File(path, filename) ); 
p.print(contents) ; 


I0.close(p); 


} 


class DirMap { 
private Hashtable t = new Hashtable(); 
private String rootpath; 
DirMap() { 
rootpath = System.getProperty("user.dir"); 
} 
DirMap(String alternateDir) { 
rootpath = alternateDir; 
} 
public void add(SourceCodeFile f){ 
String path = f.directory(); 
if(!t.containsKey(path) ) 
t.put(path, new Vector()); 


((Vector)t.get(path) ).addElement(f); 


public void writePackedFile(String fname) { 


DataOutputStream packed = I0.dosOpen(fname) ; 


try { 
packed.writeBytes("###0ld Separator:" + 
SourceCodeFile.filesep + "###\n"); 


} catch(IOException e) { 


Pr.error("Writing separator to " + fname); 


} 


Enumeration e = t.keys(); 
while(e.hasMoreElements()) { 
String dir = (String)e.nextElement(); 
System.out.printin( 
"Writing directory " + dir); 
Vector v = (Vector)t.get(dir); 
for(int i = 0; i < v.size(); i++) { 
SourceCodeFile f = 
(SourceCodeFile)v.elementAt(i); 


f.writePacked(packed); 


} 


I0.close(packed); 


} 


// Write all the files in their directories: 


public void write() { 
Enumeration e = t.keys(); 
while(e.hasMoreElements()) { 
String dir = (String)e.nextElement(); 
Vector v = (Vector)t.get(dir); 
for(int i = 0; i < v.size(); i++) { 
SourceCodeFile f = 
(SourceCodeFile)v.elementAt(i); 
f.writeFile(rootpath); 
} 
// Add file indicating file quantity 
// written to this directory as a check: 
I0.close(1I0.dosOpen( 
new File(new File(rootpath, dir), 


Integer.toString(v.size())+".files"))); 


} 


public class CodePackager { 
private static final String usageString = 
"usage: java CodePackager packedFileName" + 
"\nExtracts source code files from packed \n" + 


"version of Tjava.doc sources into " + 


"directories off current directory\n" + 
"java CodePackager packedFileName newDir\n" + 
"Extracts into directories off newDir\n" + 
"java CodePackager -p source.txt packedFile" + 
"\nCreates packed version of source files" + 
"\nfrom text version of Tjava.doc"; 
private static void usage() { 
System.err.println(usageString); 
System.exit(1); 
} 
public static void main(String[] args) { 
if(args.length == 0) usage(); 
if(args[0].equals("-p")) { 
if(args.length != 3) 
usage(); 
createPackedFile(args); 
} 
else { 
if(args.length > 2) 
usage(); 


extractPackedFile(args); 


private static String currentLine; 
private static BufferedReader in; 
private static DirMap dm; 

private static void 


createPackedFile(String[] args) { 


dm = new DirMap(); 
in = 10.disOpen(args[1]); 
try { 


while((currentLine = in.readLine() ) 
I= null) { 
if(currentLine.startswith( 
SourceCodeFile.startMarker)) { 
dm.add(new SourceCodeFile( 
CurrentLine, in)); 
} 
else if(currentLine.startswith( 
SourceCodeFile.endMarker ) ) 
Pr.error("file has no start marker"); 
// Else ignore the input line 
} 
} catch(IOException e) { 


Pr.error("Error reading " + args[1]); 


I0.close(in); 
dm.writePackedFile(args[2]); 
} 
private static void 
extractPackedFile(String[] args) { 
if(args.length == 2) // Alternate directory 
dm = new DirMap(args[1]); 
else // Current directory 
dm = new DirMap(); 

in = 1I10.disOpen(args[0]); 

String s = null; 

try { 

s = in.readLine(); 
} catch(IOException e) { 
Pr.error("Cannot read from " + in); 

} 

// Capture the separator used in the system 
// that packed the file: 
if(s.indexOf("###0ld Separator:") != -1 ) { 

String oldsep = s.substring( 
"###0ld Separator:".length()); 
oldsep = oldsep.substring( 


0, oldsep. indexOf('#')); 


SourceCodeFile.oldsep = oldsep; 
} 
SourceCodeFile sf = new SourceCodeFile(in); 
while(sf.hasFile()) { 
dm.add(sf); 
sf = new SourceCodeFile(in); 
} 
dm.write(); 
} 
} ///:~ 








我 们 注意 到 package 语 句 己 经 作为 注释 标志 出 来 了 。 由 于 这 是 本 章 的 第 
一 个 程序 ， 所 以 package 语 句 是 必需 的 ， 用 它 告诉 CodePackager 已 改换 到 
男 一 章 。 但 是 把 它 放 入 包 里 却 会 成 为 一 个 问题 。 当 我 们 创建 一 个 包 的 时 
候 ， 需 要 将 结果 程序 同一 个 特定 的 目录 结构 联系 在 一 起 ， 这 一 做 法 对 本 
书 的 大 多 数 例 子 都 是 适用 的 。 但 在 这 里 ，CodePackager 程 序 必须 在 一 个 
专用 的 目录 里 编译 和 运行 ， 所 以 package 语 句 作为 注释 标记 出 去 。 但 对 
CodePackager 来 说 ， 它 “看 起 来 ”依然 象 一 个 普通 的 package 语 句 ， 因 为 程 
序 还 不 是 特别 复杂 ， 不 能 侦 得 到 多 行 注释 〈 没 有 必要 做 得 这 么 复杂 ， 这 
里 只 要 求 方便 就 行 ) 。 


头 两 个 类 是 “ 文 持 / 工具 ”类 ， 作 用 是 使 程序 剩余 的 部 分 在 编写 时 更 加 连 
贯 ， 也 更 便于 阅读 。 第 一 个 是 Pr， 它 类 似 ANSI C 的 perror 库 ， 两 者 都 能 
打印 出 一 条 错误 提示 消息 《但 同时 也 会 退出 程序 ) 。 第 二 个 类 将 文件 的 
创建 过 程 封装 在 内 ， 这 个 过 程 已 在 第 10 间 介绍 过 了 ; 大 家 已 经 知道 ， 这 
样 做 很 快 就 会 变 得 非常 累 袭 和 麻烦 。 为 解决 这 个 问题 ， 第 10 章 提供 的 方 
案 致 力 于 新 类 的 创建 ， 但 这 儿 的 “静态 ”方法 已 经 使 用 过 了 。 在 那些 方法 
中 ， 正 常 的 违例 会 被 捕获 ， 并 相应 地 进行 处 理 。 这 些 方法 使 剩余 的 代码 
wis EIR, BAW. 


帮助 解决 问题 的 第 一 个 类 是 SourceCodeFile (源码 文件 ) ， 它 代表 本 书 




















一 个 源码 文件 包含 的 所 有 信息 (内 容 、 文 件 名 以 及 目录 ) 。 它 同时 还 包 
含 了 一 系列 String 和 常数 ， 分 别 代 表 一 个 文件 的 开始 与 结束 ; 在 打包 文件 
内 使 用 的 一 个 标记 ; 当前 系统 的 换行 符 ;， 文件 路 径 分 隔 符 《注意 要 用 
System.getProperty0 侦 碍 本 地 版 本 是 什么 ) ;以 及 一 大 段 版 权 声 明 ， 它 
是 从 下 面 这 个 Copyright.txt 文 件 里 提取 出 来 的 : 


447111111111111111111111111111111111111111111111// 








// Copyright (c) Bruce Eckel, 1998 

// Source code file from the book "Thinking in Java" 
// All rights reserved EXCEPT as allowed by the 

// following statements: You may freely use this file 
// for your own work (personal or commercial), 

// including modifications and distribution in 

// executable form only. Permission is granted to use 
// this file in classroom situations, including its 
// use in presentation materials, as long as the book 
// "Thinking in Java" is cited as the source. 

// Except in classroom situations, you may not copy 
// and distribute this code; instead, the sole 

// distribution point is http://www.BruceEckel.com 

// (and official mirror sites) where it is 

// freely available. You may not remove this 

// copyright and notice. You may not distribute 

// modified versions of the source code in this 


// package. You may not use this file in printed 


// 


// 


// 


// 


// 


// 


// 


// 


// 


if 


// 


// 


// 


// 


// 


// 


// 


// 


// 


// 


// 


// 


// 


media without the express permission of the 
author. Bruce Eckel makes no representation about 
the suitability of this software for any purpose. 
It is provided "as is" without express or implied 
warranty of any kind, including any implied 
warranty of merchantability, fitness for a 
particular purpose or non-infringement. The entire 
risk as to the quality and performance of the 
software is with you. Bruce Eckel and the 
publisher shall not be liable for any damages 
suffered by you or any third party as a result of 
using or distributing software. In no event will 
Bruce Eckel or the publisher be liable for any 
lost revenue, profit, or data, or for direct, 
indirect, special, consequential, incidental, or 
punitive damages, however caused and regardless of 
the theory of liability, arising out of the use of 
or inability to use software, even if Bruce Eckel 
and the publisher have been advised of the 
possibility of such damages. Should the software 
prove defective, you assume the cost of all 
necessary servicing, repair, or correction. If you 


think you've found an error, please email all 


// modified files with clearly commented changes to: 
// Bruce@EckelObjects.com. (please use the same 
// address for non-code errors found in the book). 


4471111111111111111111111111111111111111111111111// 


从 一 个 打包 文件 中 提取 文件 时 ， 当 初 所 用 系统 的 文件 分 隅 符 也 会 标注 出 
来 ， 以 便 用 本 地 系统 适用 的 符号 蔡 换 它 。 


当前 章 的 子 目录 保存 在 chapter 字 段 中 ， 它 初始 化 成 c02( 大 家 可 注意 一 
下 第 2 章 的 列表 正好 没有 包含 一 个 打包 语句 ) 。 只 有 在 当前 文件 里 发 现 
一 个 package (打包) 语句 时 ，chapter 字 段 才 会 发 生 改 变 。 


1. 构建 一 个 打包 文件 


第 一 个 构建 器 用 于 从 本 书 的 ASCII 文 本 版 里 提取 出 一 个 文件 。 发 出 调用 
的 代码 《在 列表 里 较 深 的 地 方 ) 会 读 入 并 检查 每 一 行 ， 直 到 找到 与 一 个 
列表 的 开头 相符 的 为 止 。 在 这 个 时 候 ， 它 束 会 新 建 一 个 SourceCodeFile 
对 象 ， 将 第 一 行 的 内 容 (已 经 由 调用 代码 读 入 了 ) 传递 给 它 ， 同 时 还 要 
传递 BufferedReader 对 象 ， 以 便 在 这 个 缓冲 区 中 提取 源 但 列表 剩余 的 内 
容 。 

从 这 时 起 ， 大 家 会 发 现 String 方 法 被 频繁 运用 。 为 提取 出 文件 名 ， 需 调 
用 substringO 的 过 载 版 本 ， 令 其 从 一 个 起 始 侦 移 开始 ， 一 直 读 到 字 串 的 
末尾 ， 从 而 形成 一 个 “子囊 ”。 为 算出 这 个 起 始 索引 ， 先 要 用 lengthO 得 出 
startMarker 的 总 长 ， 再 用 trim( 删 除 字 串 头 尾 多 余 的 空格 。 第 一 行 在 文件 
名 后 也 可 能 有 一 些 字符 ; 它们 是 用 ipdexOfO 侦 测 出 来 的 。 若 没有 发 现 找 
到 我 们 想 寻找 的 字符 ， 束 返回 -1; ATK BINS A, WORE MAK 
出 现 的 位 置 。 注 意 这 也 是 indexOfO 的 一 个 过 载 版 本 ， 采 用 一 个 字 串 作为 
参数 ， 而 非 一 个 字符 。 


解析 出 并 保存 好 文件 名 后 ， 第 一 行 会 被 置 入 字 串 contents 中 《该 字 串 用 
于 保存 源码 清单 的 完整 正文 ) 。 随 后 ， 将 剩余 的 代码 行 读 入 ， 并 合并 进 
入 contents 字 串 。 当 然 事 情 并 没有 想象 的 那么 简单 ， 因 为 特定 的 情况 需 
加 以 特别 的 控制 。 一 种 情况 是 错误 检查 : 知 直 接 遇 到 一 个 

startMarker 〈 起 始 标 记 ) ， 表 明 当 前 操作 的 这 个 代码 列表 没有 设置 一 个 




















结束 标记 。 这 属于 一 个 出 错 条 件 ， 需 要 退出 程序 。 


男 一 种 特殊 情况 与 package 关 键 字 有 关 。 尺 管 Java 是 一 种 自由 形式 的 语 
言 ， 但 这 个 程序 要 求 package 关 键 字 必 须 位 于 行 首 。 知 发 现 package 关 键 
字 ， 就 通过 检查 位 于 开头 的 空格 以 及 位 于 末尾 的 分 扎 ， 从 而 提取 出 包 名 
(注意 亦 可 一 次 单独 的 操作 实现 ， 方 法 是 使 用 过 载 的 substring0， 令 其 
同时 检查 起 始 和 结束 索引 位 置 ) 。 随 后 ， 将 包 名 中 的 点 号 蔡 换 成 特定 的 
文件 分 隔 符 一 一 当然 ， 这 里 要 假设 文件 分 隅 符 仅 有 一 个 字符 的 长 度 。 尽 
管 这 个 假设 可 能 对 目前 的 所 有 系统 都 是 适用 的 ， 但 一 旦 遇 到 问题 ， 一 定 
DRD IRA PAE. 


默认 操作 是 将 每 一 行 都 连接 到 contents 里 ， 同 时 还 有 换行 字符 ， 直 到 遇 
到 一 个 endMarker 〈 结 束 标记 ) 为 止 。 访 标记 指出 构建 器 应 当 停 止 了 。 
各 在 endMarker 之 前 遇 到 了 文件 结尾 ， 就 认为 存在 一 个 错误 。 


2. 从 打包 文件 中 提取 


第 二 个 构建 器 用 于 将 源码 文件 从 打包 文件 中 恢复 (提取 )〉 出来。 在 这 

儿 ， 作 为 调用 者 的 方法 不 必 担 心 会 跳 过 一 些 中 则 文本 。 打 包 文 件 包含 了 
所 有 源码 文件 ， 它 们 相互 间 紧 密 地 靠 在 一 起 。 需 要 传递 给 该 构建 器 的 仪 
仪 是 一 个 BufferedReader， 它 代表 着 “信息 源 "。 构 建 右 会 从 中 提取 出 自 

己 需 要 的 信息 。 但 在 每 个 代码 列表 开始 的 地 方 还 有 一 些 配 置信 息 ， 它 们 
的 身份 是 用 packMarker〈 打 包 标 记 ) 指出 的 。 知 packMarker 不 存在 ， 意 
味 着 调用 者 试图 用 错误 的 方法 来 使 用 这 个 构建 器 。 


一 旦 发 现 packMarker， 就 会 将 其 剥离 出 来 ， 并 提取 出 目录 名 (用 一 
SHE) 以 及 文件 名 《直到 行 末 ) 。 不 管 在 哪 种 情况 下 ， 旧 分 隔 符 都 
会 被 蔡 换 成 本 地 适用 的 一 个 分 隔 符 ， 这 是 用 String replace() 方 法 实现 
的 。 老 的 分 隔 符 被 置 于 打包 文件 的 开头 ， 在 代码 列表 稍 靠 后 的 一 部 分 即 
可 看 到 是 如 何 把 它 提取 出 来 的 。 


构建 器 剩 下 的 部 分 就 非常 简单 了 。 它 读 入 每 一 行 ， 把 它 合 并 到 contents 
里 ， 直 到 遇见 endMarker 为 止 。 


3. 程序 列表 的 存 取 


接 下 来 的 一 系列 方法 是 简单 的 访问 器 : directory(). filename() 〈 注 意 方 
法 可 能 与 字段 有 相同 的 拼写 和 大 小 写 形式 ) 和 contents()。 而 hasFile() 用 



































于 指出 这 个 对 象 是 人 否 包含 了 一 个 文件 “很 快 就 会 知道 为 什么 需要 这 


最 后 三 个 方法 致力 于 将 这 个 代码 列表 与 过 人 双人 要 么 通过 
writePacked() 写 入 一 个 打包 文件 ， 要 么 通过 writeFile() 写 入 一 个 ] avait hg 
文件 。writePacked0 需 要 的 唯一 东西 就 是 DataOutputStream， 它 是 在 别 的 
地 方 打开 的 ， 代 表 痢 准备 写 入 的 文件 。 它 先 把 头 信息 置 入 第 一 行 ， 再 调 
用 writeBytes() 将 contents 内容) 写成 一 种 “通用 ”格式 。 


准备 写 Java 源 码 文件 时 ， 必 须 先 把 文件 建 好 。 这 是 用 IO.psOpen0 实 现 
的 。 我 们 需要 辐 它 传递 一 个 File 对 象 ， 其 中 不 仅 包 含 了 文件 名 ， 也 包含 
了 路 径 信息 。 但 现在 的 问题 是 : 这 个 路 径 实 际 存在 吗 ? 用 户 可 能 决定 将 
所 有 源码 目录 都 置 入 一 个 完全 不 同 的 子 目 录 ， 那 个 目录 可 能 是 尚 不 存在 
的 。 所 以 在 正式 写 每 个 文件 之 前 ， 都 要 调用 File.mkdirs() 方 法 ， 建 好 我 
们 想 回 其 中 写 入 文件 的 目录 路 径 。 它 可 一 次 性 建 好 整个 路 径 。 


4. 整套 列表 的 包容 


以 子 目 录 的 形式 组 织 代 码 列表 是 非 癌 方便 的 ， 尽 管 这 要 求 先 在 内 存 中 建 
好 整套 列表 。 之 所 以 要 这 样 做 ， 还 有 男 一 个 很 有 说 服 力 的 原因 为 了 构 
建 更 “健康 ”的 系统 。 也 就 是 说 ， 在 创建 代码 列表 的 每 个 子 目录 时 ， 都 会 
加 入 一 个 额外 的 文件 ， 它 的 名 字 包 含 了 那个 目录 内 应 有 的 文件 数目 。 


DirMap 类 可 帮助 我 们 实现 这 一 效果 ， 并 有 效 地 演示 了 一 个 “多 重 映 射 ”的 
概述 。 这 是 通过 一 个 散 列 表 〈Hashtable) 实现 的 ， 它 的 “ 键 ? 是 准备 创建 
的 子 目 录 ， 而 “ 值 ”是 包含 了 那个 特定 目录 中 的 SourceCodeFile 对 象 的 
Vector 对 象 。 所 以 ， 我 们 在 这 儿 并 不 是 将 一 个 “ 键 ? 映 射 〈 或 对 应 ) 到 一 
个 值 ， 而 是 通过 对 应 的 Vector， 将 一 个 键 “ 多 重 映射 ? 到 一 系列 值 。 尽 管 
这 听 起 来 似乎 很 复杂 ， 但 具体 实现 时 却 是 非常 简单 和 直接 的 。 大 家 可 以 
看 到 ，DirMap 类 的 大 多 数 代 码 都 与 回 文件 中 的 写 入 有 关 ， 而 非 与 “多 重 
映射 >? 有 有关。 与 它 有 关 的 代码 仅 极 少数 而 已 。 


可 通过 两 种 方式 建立 一 个 DirMap“《〈 目 录 映 射 或 对 应 ) 关系 : PRU ME aR 
假定 我 们 希望 目录 从 当前 位 置 癌 下 展开 ， 而 另 一 个 构建 器 让 我 们 为 起 始 
目录 指定 一 个 备用 的 “绝对 ”路径 。 


add0) 方 法 是 一 个 采取 的 行动 比较 密集 的 场所 。 首 先 将 directory() 从 我 们 
想 添 加 的 SourceCodeFile 里 提取 出 来 ， 然 后 检查 散 列 表 (Hashtable) , 












































看 看 其 中 是 否 已 经 包含 了 那个 键 。 如 果 没 有 ， 束 同 散 列表 加 入 一 个 新 的 
Vector， 并 将 它 同 那个 键 关 联 到 一 起 。 到 这 时 ， 不 管 采取 的 是 什么 途 
径 ，Vector 都 已 经 就 位 了 ， 可 以 将 它 提取 出 来 ， 以 全 添加 
SourceCodeFile。 由 于 Vector 可 象 这 样 同 散 列 表 方 便 地 合并 到 一 起 ， 所 以 
我 们 从 两 方面 都 能 感觉 得 非常 方便 。 


写 一 个 打包 文件 时 ， 需 打开 一 个 准备 写 入 的 文件 〈 当 作 
DataOutputStream 打 开 ， 使 数据 具有 “通用 ”性 ) ， 并 在 第 一 行 写 入 与 老 
的 分 隔 符 有 关 的 头 信息 。 接 着 产生 对 Hashtable 键 的 一 个 Enumeration 〈 枚 
举 ) ， 并 通 历 其 中 ， 选 择 每 一 个 目录 ， 并 取得 与 那个 目录 有 关 的 
Vector， 使 那个 Vector 中 的 每 个 SourceCodeFile 都 能 写 入 打包 文件 中 。 


用 write0) 将 Java 源 码 文 件 写 入 它们 对 应 的 目录 时 ， 采 用 的 方法 几乎 与 
writePackedFile0 完 全 一 致 ， 因 为 两 个 方法 都 只 需 简 单调 用 
SourceCodeFile 中 适当 的 方法 。 但 在 这 里 ， 根 路 径 会 传递 给 
SourceCodeFile.writeFile()。 所 有 文件 都 写 好 后 ， 名 字 中 指定 了 已 写 文件 
数量 的 那个 附加 文件 也 会 被 写 入 。 


5. 主 程序 


前 面 介 绍 的 那些 类 都 要 在 CodePackager 中 用 到 。 大 家 首先 看 到 的 是 用 法 
字 串 。 一 旦 最 终 用 户 不 正确 地 调用 了 程序 ， 就 会 打印 出 介绍 正确 用 法 的 
这 个 字 串 。 调 用 这 个 字 串 的 是 usage(0 方 法 ， 同 时 还 要 退出 程序 。main() 
唯一 的 任务 就 是 判断 我 们 希望 创建 一 个 打包 文件 ， 还 是 希望 从 一 个 打包 
文件 中 提取 什么 东西 。 随 后 ， 它 负责 保证 使 用 的 是 正确 的 参数 ， 并 调用 
适当 的 方法 。 


创建 一 个 打包 文件 时 ， 它 默认 位 于 当前 目录 ， 所 以 我 们 用 默认 构建 器 创 
打开 文件 后 ， 其 中 的 每 一 行 都 会 读 入 ， 并 检查 是 售 符 合 特殊 
YAR 


(1) 知行 首 是 一 个 用 于 源码 列表 的 起 始 标 记 ， 束 新建 一 个 SourceCodeFile 
构建 器 会 读 入 源码 列表 剩 下 的 所 有 内 容 。 结 果 产 生 的 句柄 将 直接 
IA. DirMap. 


(2) 若 行 首 是 一 个 用 于 源码 列表 的 结束 标记 ， 表 明 某 个 地 方 出 现 错误 ， 
因为 结束 标记 应 当 只 能 由 SourceCodeFile 构 建 器 发 现 。 





























提取 / 释放 一 个 打包 文件 时 ， 提 取出 来 的 内 容 可 进入 当前 目录 ， 亦 可 进 
入 另 一 个 备用 目录 。 所 以 需要 相应 地 创建 DirMap 对 象 。 打 开 文 件 ， 并 将 
第 一 行 读 入 。 老 的 文件 路 径 分 隔 符 信息 将 从 这 一 行 中 提取 出 来 。 随 后 根 
据 输入 来 创建 第 一 个 SourceCodeFile 对 象 ， 它 会 加 入 DirMap。 只 要 包含 
了 一 个 文件 ， 新 的 SourceCodeFile 对 象 就 会 创建 并 加 入 “〈 创 建 的 最 后 一 
个 用 光 输 入 内 容 后 ， 会 简单 地 返回 ， 然 后 hasFile() 会 返回 一 个 错误 ) 。 


17.1.2 检查 大 小 写 样式 


尽管 对 涉及 文字 人 处理 的 一 些 项 目 来 说 ， 前 例 显得 比较 方便 ， 但 下 面 要 介 
绍 的 项 目 却 能 立即 发 挥 作用 ， 因 为 它 执 行 的 是 一 个 样式 检查 ， 以 确保 我 
们 的 大 小 写 形式 符合 “事实 上 ”的 Java 样 式 标 准 。 它 会 在 当前 目录 中 打开 
每 个 java 文件 ， 并 提取 出 所 有 类 名 以 及 标识 符 。 各 发现 有 不 符合 Java 样 
式 的 情况 ， 残 向 我 们 提出 报告 。 


为 了 让 这 个 程序 正确 运行 ， 首 先 必须 构建 一 个 类 名 ， 将 它 作 为 一 个 “ 仓 
库 ”， 负 责 容 纳 标 准 Java 库 中 的 所 有 类 名 。 为 达到 这 个 目的 ， 需 遍历 用 
于 标准 Java 库 的 所 有 源码 子 目 录 ， 并 在 每 个 子 目 录 都 运行 ClassScanner。 
至 于 参数 ， 则 提供 仓库 文件 的 名 字 “每 次 都 用 相同 的 路 径 和 名 字 ) 和 命 
令 行 开 关 -a， 指 出 类 名 应 当 添 加 到 该 仓库 文件 中 。 


为 了 用 程序 检查 自己 的 代码 ， 需 要 运行 它 ， 并 向 它 传递 要 使 用 的 仓库 文 
件 的 路 径 与 名 字 。 它 会 检查 当前 目录 中 的 所 有 类 和 标识 符 ， 并 告诉 我 们 
哪些 没有 遵守 典型 的 Java 大 写 写 规范 。 

要 注意 这 个 程序 并 不 是 十 全 十 美的 。 有 些 时 候 ， 它 可 能 报告 自己 查 到 一 
个 问题 。 但 当 我 们 仔细 检查 代码 的 时 候 ， 却 发 现 没 有 什么 需要 更 改 的 。 
尽管 这 有 点 儿 烦 人 ， 但 仍 比 自己 动手 检查 代码 中 的 所 有 错误 强 得 多 。 
下 面 列 出 源 人 代码， 后面 有 详细 的 解释 : 


//: ClassScanner.java 



































// Scans all files in directory for classes 
// and identifiers, to check capitalization. 


// Assumes properly compiling code listings. 


// Doesn't do everything right, but is a very 
// useful aid. 
import java.io.*; 
import java.util.*; 
class MultiStringMap extends Hashtable { 
public void add(String key, String value) { 
if(!containsKey(key) ) 
put(key, new Vector()); 
((Vector )get(key) ).addElement (value); 
} 
public Vector getVector(String key) { 
if(!containsKey(key)) { 
System.err.printin( 
"ERROR: can't find key: " + key); 
System.exit(1); 
} 
return (Vector)get(key); 
} 
public void printValues(PrintStream p) { 
Enumeration k = keys(); 
while(k.hasMoreElements()) { 
String oneKey = (String)k.nextElement(); 


Vector val = getVector(oneKey); 


for(int i = 0; i < val.size(); i++) 


p.printin((String)val.elementAt(1)); 


} 


public class ClassScanner { 
private File path; 
private String[] fileList; 
private Properties classes = new Properties(); 
private MultiStringMap 
classMap = new MultiStringMap(), 
identMap = new MultiStringMap(); 
private StreamTokenizer in; 
public ClassScanner() { 
path = new File("."); 
fileList = path.list(new JavaFilter()); 
for(int i = 0; i < fileList.length; i++) { 
System.out.println(fileList[i]); 


scanListing(fileList[i]); 


} 


void scanListing(String fname) { 


try { 


in = new StreamTokenizer ( 
new BufferedReader ( 
new FileReader(fname) )); 
// Doesn't seem to work: 

// in.slashStarComments(true); 

// in.SlashSlashComments(true) ; 
in.ordinaryChar('/'); 
in.ordinaryChar('.'); 
in.wordChars('_', '_'); 
in.eolIsSignificant(true); 
while(in.nextToken() != 

StreamTokenizer.TT_EOF) { 
if(in.ttype == '/') 
eatComments(); 
else if(in.ttype == 
StreamTokenizer.TT_WORD) { 
if(in.sval.equals("class") | | 
in.sval.equals("interface")) { 
// Get class name: 
while(in.nextToken() != 
StreamTokenizer .TT_EOF 
&& in.ttype != 


StreamTokenizer .TT_WORD) 


classes.put(in.sval, in.sval); 
CclassMap.add(fname, in.sval); 
} 
if(in.sval.equals("import") || 
in.sval.equals("package" ) ) 
discardLine(); 
else // It's an identifier or keyword 


identMap.add(fname, in.sval); 


} 


} catch(IOException e) { 


e.printStackTrace(); 


} 


void discardLine() { 
try { 
while(in.nextToken() != 
StreamTokenizer .TT_EOF 
&& in.ttype != 
StreamTokenizer .TT_EOL) 
; // Throw away tokens to end of line 


} catch(IOException e) { 


e.printStackTrace(); 


} 


// StreamTokenizer's comment removal seemed 
// to be broken. This extracts them: 
void eatComments() { 

try { 
if(in.nextToken() != 
StreamTokenizer.TT_EOF) { 
if(in.ttype == '/') 
discardLine(); 
else if(in.ttype != '*') 
in.pushBack(); 
else 
while(true) { 
if(in.nextToken() == 
StreamTokenizer .TT_EOF) 
break; 
if(in.ttype == '*') 
if(in.nextToken() != 
StreamTokenizer .TT_EOF 
&& in.ttype == '/') 


break; 


} 


} catch(IOException e) { 


e.printStackTrace(); 


} 


public String[] classNames() { 
String[] result = new String[classes.size()]; 
Enumeration e = classes.keys(); 
int i = 0; 
while(e.hasMoreElements() ) 
result[i++] = (String)e.nextElement(); 
return result; 
} 
public void checkClassNames() { 
Enumeration files = classMap.keys(); 
while(files.hasMoreElements()) { 
String file = (String)files.nextElement(); 
Vector cls = classMap.getVector(file); 
for(int i = 0; i < cls.size(); i++) { 
String className = 
(String)cls.elementAt(1); 


if (Character .isLowerCase( 


className.charAt(0))) 
System.out.printin( 

"Class capitalization error, file: " 

+ file + ", class: " 


+ className); 


j 


public void checkIdentNames() { 
Enumeration files = identMap.keys(); 
Vector reportSet = new Vector(); 
while(files.hasMoreElements()) { 
String file = (String)files.nextElement(); 
Vector ids = identMap.getVector(file); 
for(int i = 0; i < ids.size(); i++) { 
String id = 
(String)ids.elementAt(1i); 
if(!classes.contains(id)) { 
// Ignore identifiers of length 3 or 
// longer that are all uppercase 
// (probably static final values): 
if(id.length() >= 3 && 


id.equals( 


id.toUpperCase())) 
continue; 
// Check to see if first char is upper: 
if (Character .isUpperCase(id.charAt(0))){ 
if(reportSet.indexOf(file + id) 
== -1){ // Not reported yet 
reportSet.addElement(file + id); 
System. out.printin( 
"Ident capitalization error in:" 


+ file + ", ident: " + id); 


} 


static final String usage = 
"Usage: \n" + 
"ClassScanner classnames -a\n" + 
"\tAdds all the class names in this \n" + 
"\tdirectory to the repository file \n" + 
"\tcalled 'classnames'\n" + 


"ClassScanner classnames\n" + 


"\tChecks all the java files in this \n" + 
"\tdirectory for capitalization errors, \n" + 
"\tusing the repository file 'classnames'"; 
private static void usage() { 
System.err.println(usage); 
System.exit(1); 
} 
public static void main(String[] args) { 
if(args.length < 1 || args.length > 2) 
usage(); 
ClassScanner c = new ClassScanner()j; 
File old = new File(args[0]); 
if(old.exists()) { 
try { 
// Try to open an existing 
// properties file: 
InputStream oldlist = 
new BufferedInputStream( 
new FileInputStream(old) ); 
c.classes.load(oldlist); 
oldlist.close(); 
} catch(IOException e) { 


System.err.printin("Could not open " 


+ old + " for reading"); 


System.exit(1); 


} 
if(args.length == 1) { 
c.checkClassNames(); 
c.checkIdentNames(); 
} 
// Write the class names to a repository: 
if(args.length == 2) { 
if(!args[1].equals("-a")) 
usage(); 
try { 
BufferedoutputStream out = 
new BufferedOutputStream( 
new FileOutputStream(args[0])); 
c.classes.save(out, 
"Classes found by ClassScanner.java"); 
out.close(); 
} catch(I0Exception e) { 
System.err.printin( 
"Could not write " + args[0]); 


System.exit(1); 


} 
class JavaFilter implements FilenameFilter { 
public boolean accept(File dir, String name) { 
// Strip path information: 
String f = new File(name).getName(); 
return f.trim().endswith(".java"); 


l 


} ///:~ 


MultistringMap 类 是 个 特殊 的 工具 ， 人 允许 我 们 将 一 组 字 串 与 每 个 键 项 对 
应 《映射 ) 起 来 。 和 前 例 一 样 ， 这 里 也 使 用 了 一 个 散 列 表 

(Hashtable) ， 不 过 这 次 设置 了 继承 。 访 散 列表 将 键 作 为 映射 成 为 
Vector 值 的 单一 的 字 串 对 待 。add0) 方 法 的 作用 很 简单 ， 负 责 检查 散 列 表 
里 是 否 存在 一 个 键 。 如 果 不 存 在 ， 束 在 其 中 放置 一 个 。getVector() 方 法 
为 一 个 特定 的 键 产 生 一 个 Vector; 而 printValuesO 将 所 有 值 逐 个 Vector 地 
打印 出 来 ， 这 对 程序 的 调试 非常 有 用 。 


为 简化 程序 ， 来 自 标 准 Java 库 的 类 名 全 都 置 入 一 个 Properties (属性 ) 对 
象 中 (来 自 标 准 Java 库 ) 。 记 住 Properties 对 象 实际 是 个 散 列 表 ， 其 中 只 
容纳 了 用 于 键 和 值 项 的 String 对 象 。 然 而 仅 需 一 次 方法 调用 ， 我 们 即 可 
把 它 保存 到 磁盘 ， 或 者 从 磁盘 中 恢复 。 实 际 上 ， 我 们 只 需要 一 个 名 字 列 
表 ， 所 以 为 键 和 值 都 使 用 了 相同 的 对 象 。 


针对 特定 目录 中 的 文件 ， 为 找 出 相应 的 类 与 标识 符 ， 我 们 使 用 了 两 个 
MultiStringMap: classMap 以 及 identMap。 此 外 在 程序 启动 的 时 候 ， 它 会 
将 标准 类 名 仓库 装载 到 名 为 classes 的 Properties 对 象 中 。 一 旦 在 本 地 目录 
发 现 了 一 个 新 类 名 ， 也 会 将 其 加 入 classes 以 及 classMap 。 这 样 一 来 ， 























classMap 束 可 用 于 在 本 地 目录 的 所 有 类 间 授 历 ， 而 且 可 用 classes 检 查 当 
前 标记 是 不 是 一 个 类 名 〔 它 标记 着 对 象 或 方法 定义 的 开始 ， 所 以 收集 接 
下 去 的 记号 一 一 直到 们 到 一 个 分 号 一 一 并 将 它们 都 置 入 identMap) 。 


ClassScanner 的 默认 构建 右 会 创建 一 个 由 文件 名 构成 的 列表 (采用 
FilenameFilter 的 JavaFilter 实 现形 式 ， 参 见 第 10 章 ) 。 随 后 会 为 每 个 文件 
名 都 调用 scanListing()。 


在 scanListing() 内 部 ， 会 打开 源码 文件 ， 并 将 其 转换 成 一 个 
StreamTokenizer。 根 据 Java 帮 助 文档 ， 将 true 传 递 给 slashStartComments() 
和 slashSlashComments() 的 本 意 应 当 是 剥 除 那些 注释 内 容 ， 但 这 样 做 似乎 
有 些 问 题 (在 Java 1.0 中 几乎 无 效 ) 。 所 以 相反 ， 那 些 行 被 当 作 注释 标 
记 出 去 ， 并 用 另 一 个 方法 来 提取 注释 。 为 达到 这 个 目的 ，/ 必 须 作 为 一 
个 原始 字符 捕获 ， 而 不 是 让 StreaamTokeinzer 将 其 当 作 注释 的 一 部 分 对 
待 。 此 时 要 用 ordinaryChar() 方 法 指示 StreamTokenizer 采 取 正 确 的 操作 。 
同样 的 道理 也 适用 于 点 号 〈("') ， 因 为 我 们 希望 让 方法 调用 分 离 出 单独 
的 标识 符 。 但 对 下 划 线 来 说 ， 它 最 初 是 被 StreamTokenizer 当 作 一 个 单独 
的 字符 对 竺 的， 但 此 时 应 把 它 留 作 标识 符 的 一 部 分 ， 因 为 它 在 static final 
值 中 以 TT_EOF 等 等 形式 使 用 。 当 然 ， 这 一 点 只 对 目前 这 个 特殊 的 程序 
成 芯 。wordChars0) 方 法 需要 取得 我 们 想 添 加 的 一 系列 字符 ， 把 它们 留 在 
作为 一 个 单词 看 待 的 记号 中 。 最 后 ， 在 解析 单行 注释 或 者 放弃 一 行 的 时 
候 ， 我 们 需要 知道 一 个 换行 动作 什么 时 候 发 生 。 上 所 以 通过 调用 
eollsSignificant(true)， 换 行 符 〈EOL ) 会 被 显示 出 来 ， 而 不 是 被 
StreamTokenizer 吸 收 。 


scanListing() 剩 余 的 部 分 将 读 入 和 检查 记 写 ， 直 至 文件 尾 。 一 旦 
nextToken() 返 回 一 个 final static 值 一 一 StreamTokenizer.TT_EOF， 就 标志 
着 已 经 抵达 文件 尾部 。 


若 记 号 是 个 /， 意 味 着 它 可 能 是 个 注释 ， 所 以 就 调用 eatCommentsO0， 对 
这 种 情况 进行 处 理 。 我 们 在 这 儿 唯 一 感 兴趣 的 其 他 情况 是 它 是 否 为 一 个 
单词 ， 当 然 还 可 能 存在 另 一 些 特 殊 情 况 。 


如 果 单 词 是 class〈 类 ) 或 interface 〈 接 口 ) ， 那 么 接着 的 记号 就 应 当代 
表 一 个 类 或 接口 名 字 ， 并 将 其 置 入 classes 和 classMap。 知 单词 是 import 
或 者 package， 那 么 我 们 对 这 一 行 剩 下 的 东西 就 没什么 兴趣 了 。 其 他 所 
有 东西 肯定 是 一 个 标识 符 ( 这 是 我 们 感 兴趣 的 ) ， 或 者 是 一 个 关键 字 
《对 此 不 感 兴趣 ， 但 它们 采用 的 肯定 是 小 写 形 式 ， 所 以 不 必 兴 师 动 众 地 









































检查 它们 〉 。 它 们 将 加 入 到 identMap。 


discardLine() 方 法 是 一 个 简单 的 工具 ， 用 于 查找 行 末 位 置 。 注 意 每 次 得 
到 一 个 新 记号 时 ， 都 必须 检查 行 末 。 


只 要 在 主 解 析 循 环 中 全 到 一 个 正 斜 枉 ， 束 会 调用 eatComments0) 方 法 。 然 
而 ， 这 并 不 表示 肯定 过 到 了 一 条 注释 ， 所 以 必须 将 接着 的 记号 提取 出 
来 ， 检 查 它 是 一 个 正 斜 村 (那么 这 一 行 会 被 丢弃) ， 还 是 一 个 星 号 。 但 
假如 两 者 都 不 是 ， 意 味 着 必须 在 主 解析 循环 中 将 刚才 取出 的 记号 送 回 
E! 季 运 的 是 ，pushBack(0) 方 法 允许 我 们 将 当前 记号 * 压 回 ?” 输 入 数据 
eee ne 的 时 候 ， 它 能 正确 地 得 到 刚才 送 
jE] > 和 东西。 


为 方便 起 见 ，classNames() 方 法 产生 了 一 个 数组 ， 其 中 包含 了 classes 集 合 
中 的 所 有 名 字 。 这 个 方法 未 在 程序 中 使 用 ， 但 对 代码 的 调试 非常 有 用 。 


接 下 来 的 两 个 方法 是 实际 进行 检查 的 地 方 。 在 checkClassNames() 中 ， 类 
名 从 classMap 提 取出 来 (请 记 住 ，classMap 只 包含 了 这 个 目录 内 的 名 
字 ， 它 们 按 文件 名 组 织 ， 所 以 文件 名 可 能 伴随 错误 的 类 名 打印 出 来 〉。 
为 做 到 这 一 点 ， 需 要 取出 每 个 关联 的 Vector， 并 人 遍历 其 中 ， 检 查 第 一 个 
字符 是 否 为 小 写 。 知 确实 为 小 号 ， 则 打印 出 相应 的 出 错 提示 消息 。 


在 checkIdentNames0 中 ， 我 们 采用 了 一 种 类 似 的 方法 : 每 个 标识 符 名 字 
都 从 identMap 中 提取 出 来 。 如 果 名 字 不 在 classes 列 表 中 ， 就 认为 它 是 一 
个 标识 符 或 者 关键 字 。 此 时 会 检查 一 种 特殊 情况 ， 如 果 标 识 符 的 长 度 等 
于 3 或 者 更 长 ， 而 且 所 有 字符 都 是 大 写 的 ， 则 忽略 此 标识 符 ， 因 为 它 可 
能 是 一 个 static ”final 值 ， 比 如 TT_EOF。 当 然 ， 这 并 不 是 一 种 完美 的 算 
法 ， 但 它 假 定 我 们 最 终 会 注意 到 任何 全 大 写 标识 符 都 是 不 合适 的 。 


这 个 方法 并 不 是 报告 每 一 个 以 大 与 字符 开 头 的 标识 符 ， 而 是 跟踪 那些 已 
在 一 个 名 为 reportSetO 的 Vector 中 报告 过 的 。 它 将 Vector 当 作 一 个 “ 集 
合 ” 对 符 ， 告 诉 我 们 一 个 项 目 是 否 已 在 那个 集合 中 。 该 项 目 是 通过 将 文 
和 标识 符 连 搂 起 来 生成 的 。 若 元 素 不 在 集合 中 ， 就 加 入 它 ， 然 后 产 


报告 。 
































程序 列表 剩 下 的 部 分 由 main0 构 成 ， 它 负责 控制 命令 行 参数 ， 并 判断 我 
们 是 准备 在 标准 Java 库 的 基础 上 构建 由 一 系列 类 名 构成 的 “仓库 ”， 还 是 
想 检查 已 写 好 的 那些 代码 的 正确 性 。 不 管 在 哪 种 情况 下 ， 都 会 创建 一 个 


ClassScanner 对 象 。 


无 论 准备 构建 一 个 “仓库 ”， 还 是 准备 使 用 一 个 现成 的 ， 都 必须 尝试 打开 
现 有 仓库 。 通 过 创建 一 个 File 对 象 并 测试 是 否 存在 ， 就 可 决定 是 否 打 开 
文件 并 在 ClassScanner 中 装载 classes 这 个 Properties 列 表 〈 使 用 loadO0 ) 。 

来 自 仓库 的 类 将 追加 到 由 ClassScanner 构 建 器 发 现 的 类 后 面 ， 而 不 是 将 
其 窗 盖 。 如 果 仪 提供 一 个 命令 行 参数 ， 就 意味 着 上 自己 想 对 类 名 和 标识 符 
名 字 进 行 一 次 检查 。 但 假如 提供 两 个 参数 (第 二 个 是 "-a") ， 就 表明 上 自 
己 想 构成 一 个 类 名 仓库 。 在 这 种 情况 下 ， 需 要 打开 一 个 输出 文件 ， 并 用 
Properties.save() 方 法 将 列表 写 入 一 个 文件 ， 同 时 用 一 个 字 串 提供 文件 头 
兰 自 


Huo 














17.2 方法 人 查找 工 具 


第 11 半 介绍 了 Java 1.1 新 的 “反射 ?概念 ， 并 利用 这 个 概念 查询 一 个 特定 类 
的 方法 一 一 要 么 是 由 所 有 方法 构成 的 一 个 完整 列表 ， 要 么 是 这 个 列表 的 
一 个 子 集 (名 字 与 我 们 指定 的 关键 字 相 答 〉。 那 个 例子 最 大 的 好 处 就 古 
能 目 动 显 示 出 所 有 方法 ， 不 强迫 我 们 在 继承 结构 中 遍历 ， 检 查 每 一 级 的 
基础 类 。 所 以 ， 它 实际 是 我 们 节省 编程 时 间 的 一 个 有 效 工 具 : 因为 大 多 
数 Java 方 法 的 名 字 都 规定 得 非常 全 面 和 详尽 ， 所 以 能 有 效 地 找 出 那些 包 
含 了 一 个 特殊 关键 字 的 方法 名 。 大 找到 符合 标准 的 一 个 名 字 ， 便 可 根据 
它 直 接 查 阅 联机 帮助 文档 。 

但 第 11 的 那个 例子 也 有 人 缺陷 ， 它 没有 使 用 AWT， 仅 是 一 个 纯 命 令 行 的 
应 用 。 在 这 儿 ， 我 们 准备 制作 一 个 改进 的 GUI 版 本 ， 能 在 我 们 键入 字符 
的 时 候 自动 刷新 输出 ， 也 允许 我 们 在 输出 结果 中 进行 剪 切 和 粘贴 操作 : 


//: DisplayMethods.java 








// Display the methods of any class inside 
// a window. Dynamically narrows your search. 
import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
import java.lang.reflect.*; 
import java.io.*; 
public class DisplayMethods extends Applet { 
Class cl; 
Method[] m; 


Constructor[] ctor; 


String[] n = new String[0]; 
TextField 
name = new TextField(40), 
searchFor = new TextField(30); 
Checkbox strip = 
new Checkbox("Strip Qualifiers"); 
TextArea results = new TextArea(40, 65); 
public void init() { 
strip.setState(true); 
name.addTextListener(new NameL()); 
searchFor.addTextListener(new SearchForL()); 
strip.addItemListener(new StripL()); 
Panel 
top = new Panel(), 
lower = new Panel(), 
p = new Panel(); 
top.add(new Label("Qualified class name:")); 
top.add(name) ; 
lower .add( 
new Label("String to search for:")); 
lower .add(searchFor ); 
lower.add(strip); 


p.setLayout(new BorderLayout()); 


p.add(top, BorderLayout.NORTH); 
p.add(lower, BorderLayout.SOUTH) ; 
setLayout(new BorderLayout()); 
add(p, BorderLayout.NORTH) ; 
add(results, BorderLayout.CENTER) ; 
} 
class NameL implements TextListener { 
public void textValueChanged(TextEvent e) { 
String nm = name.getText().trim(); 
if(nm.length() == 0) { 
results.setText("No match"); 
n = new String[0]; 


return; 


cl = Class.forName(nm); 

} catch (ClassNotFoundException ex) { 
results.setText("No match"); 
return; 

} 

m = cl.getMethods(); 

ctor = cl.getConstructors(); 


// Convert to an array of Strings: 


n = new String[m.length + ctor.length]; 
for(int i = 0; i < m.length; i++) 
n[i] = m[i].toString(); 
for(int i = 0; i < ctor.length; i++) 
n[i + m.length] = ctor[i].toString(); 


reDisplay(); 


} 


void reDisplay() { 
// Create the result set: 
String[] rs = new String[n.length]; 
String find = searchFor.getText(); 
int j = 0; 
// Select from the list if find exists: 
for (int i = 0; i < n.length; i++) { 
if(find == null) 
rs[j++] = n[i]; 
else if(n[i].indexOf(find) != -1) 
rs[j++] = n[il; 
} 
results.setText(""); 
if(strip.getState() == true) 


for (int i = 0; i < j; i++) 


results.append( 
StripQualifiers.strip(rs[i]) + "\n"); 
else // Leave qualifiers on 
for (int i = 0; i < j; i++) 
results.append(rs[i] + "\n"); 
} 
Class StripL implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 


reDisplay(); 


} 


Class SearchForL implements TextListener { 
public void textValueChanged(TextEvent e) { 


reDisplay(); 


} 


public static void main(String[] args) { 
DisplayMethods applet = new DisplayMethods(); 
Frame aFrame = new Frame("Display Methods"); 
aFrame.addWindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 


System.exit(0); 


} 
}); 


aFrame.add(applet, BorderLayout.CENTER); 
aFrame.setSize(500, 750); 

applet.init(); 

applet.start(); 


aFrame.setVisible(true); 


} 


class StripQualifiers { 
private StreamTokenizer st; 
public StripQualifiers(String qualified) { 
st = new StreamTokenizer ( 
new StringReader (qualified) ); 
st.ordinaryChar(' '); 
} 
public String getNext() { 
String s = null; 
try { 
if(st.nextToken() != 
StreamTokenizer.TT_EOF) { 
Switch(st.ttype) { 


case StreamTokenizer.TT_EOL: 


s = null; 
break; 
case StreamTokenizer.TT_NUMBER: 
s = Double.toString(st.nval); 
break; 
case StreamTokenizer.TT_WORD: 
S = new String(st.sval); 
break; 
default: // single character in ttype 


s = String.valueOf((char)st.ttype); 


} 


} catch(IOException e) { 
System.out.println(e); 
} 
return s; 
} 
public static String strip(String qualified) { 
StripQualifiers sq = 
new StripQualifiers(qualified); 
String s = "", si; 
while((si = sq.getNext()) != null) { 


int lastDot = si.lastIndexOf('.'); 


if(lastDot != -1) 
Si = si.substring(lastDot + 1); 


S$ += Si} 


return s; 


} 
} ///:~ 


程序 中 的 有 些 东 西 已 在 以 前 见识 过 了 。 和 本 书 的 许多 GUI 程 序 一 样 ， 这 
既 可 作为 一 个 独立 的 应 用 程序 使 用 ， 亦 可 作为 一 个 程序 片 (Applet) 使 
用 。 此 外 ，StripQualifiers 类 与 它 在 第 11 章 的 表现 是 完全 一 样 的 。 


GUI 包含 了 一 个 名 为 name 的 “文本 字段 ” (TextField) ， 或 在 其 中 输入 想 
查找 的 类 名 ; 还 包含 了 另 一 个 文本 字段 ， 名 为 searchFor， 可 选择 性 地 在 
其 中 输入 一 定 的 文字 ， 和 希望 在 方法 列表 中 得 找 那些 文字 。Checkbox 〈 复 
WERE) 允许 我 们 指出 最 终 希 望 在 输出 中 使 用 完整 的 名 字 ， 还 是 将 前 面 的 
ene 最 后 ， 结 果 显 示 于 一 个 “文本 区 域 ”(TextArea ) 











大 家 会 注意 到 这 个 程序 未 使 用 任何 按钮 或 其 他 组 件 ， 不 能 用 它们 开始 一 
次 搜索 。 这 是 由 于 无 论文 本 字段 还 是 复 选 框 都 会 受到 它们 的 “ 侦 听 者 
(Listener) 对 象 的 监视 。 只 要 作出 一 项 改变 ， 绪 果 列 表 便 会 立即 更 
新 。 知 改变 J 了 name 字段 中 的 文字 ， 新 的 文字 就 会 在 NameL 类 中 捕获 。 寿 
文字 不 为 空 ， 则 在 Class.forName0O 中 用 于 党 试 查 找 类 。 当 然 ， 在 文字 键 
入 期 间 ， 名 字 可 能 会 变 得 不 完整 ， 而 Class.forName() 会 失败 ， 这 意味 着 
它 会 “ 掷 ? 出 一 个 违例 。 该 违例 会 被 捕获 ，TextArea 会 随 之 设 

为 “Nomatch”( 没 有 相符 ) 。 但 只 要 键入 了 一 个 正确 的 名 字 〈 大 小 写 也 
TEA) , Class.forName()#t2 ht, fil getMethods()#l getConstructors() 
会 分 别 返 回 由 Method 和 Constructor 对 象 构成 的 一 个 数组 。 这 些 数 组 中 的 
每 个 对 象 都 会 通过 toString0 转 变 成 一 个 字 串 〈 这 样 便 产 生 了 完整 的 方法 
或 构建 器 签名 ) ， 而 且 两 个 列表 都 会 合并 到 n 中 一 个 独立 的 字 串 数 
组 。 数 组 n 属 于 DisplayMethods 类 的 一 名 成 员 ， 并 在 调用 reDisplayO 时 用 
于 显示 的 更 新 。 














在 改变 了 Checkbox 或 searchFor 组 件 ， 它 们 的 “ 侦 听 者 ”会 简单 地 调用 
reDisplayO。reDisplay0O 会 创建 一 个 临时 数组 ， 其 中 包含 了 名 为 rs 的 字 串 
(rs 代表 “结果 集 ” Result Set) 。 结 果 集 要 么 直接 从 n 复 制 ( 没 有 find 
关键 字 ) ， 要 么 选择 性 地 从 包含 了 find 关 键 字 的 n 中 的 字 串 复制 。 最 后 会 
检查 strip Checkbox， 看 看 用 户 是 不 是 希望 将 名 字 中 多 余 的 部 分 删除 ( 默 
WAS”) 。 寿 答案 是 肯定 的 ， 则 用 StripQualifiers.strip() 做 这 件 事 情 ; 
反之 ， 束 将 列表 简单 地 显示 出 来 。 


在 init0 中 ， 大 家 也 许 认 为 在 设置 布局 时 需要 进行 大 量 繁重 的 工作 。 事 实 
上 ， 组 件 的 布置 完全 可 能 只 需要 极 少 的 工作 。 但 象 这 样 使 用 
BorderLayout 的 好 处 是 它 允 许 用 户 改 变 窗口 的 大 小 ， 并 特别 能 使 
TextArea (XEKE) 更 大 一 些 ， 这 意味 着 我 们 可 以 改变 大 小 ， 以 便 考 
需 深 动 即 可 看 到 更 长 的 名 字 。 


编程 时 ， 大 家 会 及 现 特别 有 必要 让 这 个 工具 处 于 运行 状态 ， 因 为 在 试图 
判断 要 调用 什么 方法 的 时 候 ， 它 提供 了 最 好 的 方法 之 一 。 

















17.3 复杂 性 理论 


下 面 要 介绍 的 程序 的 前 身 是 由 Larry ”O'Brien 原创 的 一 些 代码 ， 并 以 由 
Craig Reynolds 于 1986 年 编制 的 *Boids” 程 序 为 基础 ， 当 时 是 为 了 演示 复 
杂 性 理论 的 一 个 特殊 问题 ， 名 为 “凸显 ”(Emergence) 。 


这 儿 要 达到 的 目标 是 通过 为 每 种 动物 都 规定 少许 简单 的 规则 ， 从 而 逼真 
地 再 现 动物 的 群 聚 行为 。 每 个 动物 都 能 看 到 看 到 整个 环境 以 及 环境 中 的 
其 他 动物 ， 但 它 只 与 一 系列 附近 的 “ 群 聚 伙伴 2 打交道。 动物 的 移动 基于 
三 个 简单 的 引导 行为 : 


(1) 分 隅 : 避免 本 地 群 聚 伙伴 过 于 拥挤 。 

(2) 方 回 : 章 从 本 地 和 群 聚 伙伴 的 普 过 方 癌 。 

ORE: 明 本 地 和 群 聚 伙伴 组 的 中 心 移动 。 

更 复杂 的 模型 甚至 可 以 包括 障碍 物 的 因素 ， 动 物 能 预知 和 避免 与 障碍 冲 
突 的 能 力 ， 所 以 它们 能 围 六 环 境 中 的 固定 物体 自由 活动 。 除 此 以 外 ， 动 


物 也 可 能 有 自己 的 特殊 目标 ， 这 也 许 会 造成 群体 按 特 定 的 路 径 前 进 。 为 
I 避免 障 但 以 及 目标 搜寻 的 因素 并 未 包括 到 这 里 建立 的 模型 

















尽管 计算 机 本 吴 比 较 简 陋 ， 而 且 采 用 的 规则 也 相当 简单 ， 但 结果 看 起 来 
征 真 实 的 。 也 就 是 说 ， 相 当 逼 中 的 行为 从 这 个 简单 的 模型 中 “凸显 ?出 来 
Se 

程序 以 合成 到 一 起 的 应 用 程序 / 程序 片 的 形式 提供 : 


//: FieldOBeasts.java 





// Demonstration of complexity theory; simulates 
// herding behavior in animals. Adapted from 


// a program by Larry O'Brien lobrien@msn.com 


import java.awt.*; 
import java.awt.event.*; 
import java.applet.*; 
import java.util.*; 


class Beast { 


int 
X, Y, // Screen position 
currentSpeed; // Pixels per second 


float currentDirection; // Radians 


Color color; // Fill color 


FieldOBeasts field; // Where the Beast roams 


static final int GSIZE = 10; // Graphic size 


public Beast(FieldOBeasts f, int x, int y, 
float cD, int cS, Color c) { 
field = f; 


this.x = x; 


this.y y; 
currentDirection = cD; 
currentSpeed = cS; 
color = C; 

} 

public void step() { 


// You move based on those within your 


sight: 


Vector seen = field.beastListInSector(this); 
// If you're not out in front 
if(seen.size() > 0) { 
// Gather data on those you see 
int totalSpeed = 0; 
float totalBearing = 0.0f; 
float distanceToNearest = 100000.0f; 
Beast nearestBeast = 
(Beast )Sseen.elementAt(0); 
Enumeration e = seen.elements(); 
while(e.hasMoreElements()) { 
Beast aBeast = (Beast) e.nextElement(); 
totalSpeed += aBeast.currentSpeed; 
float bearing = 
aBeast .bearingFromPointAlongAxis ( 
x, y, CurrentDirection); 
totalBearing += bearing; 
float distanceToBeast = 
aBeast.distanceFromPoint(x, y); 
if(distanceToBeast < distanceToNearest) { 
nearestBeast = aBeast; 


distanceToNearest = distanceToBeast; 


} 


// Rule 1: Match average Speed of those 
// in the list: 
currentSpeed = totalSpeed / seen.size(); 
// Rule 2: Move towards the perceived 
// center of gravity of the herd: 
currentDirection = 
totalBearing / seen.size(); 
// Rule 3: Maintain a minimum distance 
// from those around you: 
if(distanceToNearest <= 
field.minimumDistance) { 
currentDirection = 
nearestBeast.currentDirection; 
CcurrentSpeed = nearestBeast.currentSpeed; 
if(currentSpeed > field.maxSpeed) { 


CcurrentSpeed = field.maxSpeed; 


} 


else { // You are in front, so slow down 
currentSpeed = 


(int)(currentSpeed * field.decayRate); 


} 


// Make the beast move: 
x += (int)(Math.cos(currentDirection) 
* currentSpeed); 
y += (int)(Math.sin(currentDirection) 
* currentSpeed); 
x %= field.xExtent; 
y %= field.yExtent; 
if(x < 0) 
x += field.xExtent; 
if(y < 0) 
y += field.yExtent; 
} 
public float bearingFromPointAlongAxis ( 
int originX, int originY, float axis) { 
// Returns bearing angle of the current Beast 
// in the world coordiante system 
try { 
double bearingInRadians = 
Math.atan( 
(this.y - originY) / 
(this.x - originXx)); 


// Inverse tan has two solutions, so you 


// have to correct for other quarters: 
if(x < originx) { 
if(y < originyY) { 
bearingInRadians += - (float)Math.PI; 
} 
else { 
bearingInRadians = 


(float)Math.PI - bearingInRadians; 


} 

// Just subtract the axis (in radians): 
return (float) (axis - bearingInRadians); 

} catch(ArithmeticException aE) { 

// Divide by © error possible on this 

if(x > originx) { 
return 0; 
} 
else 


return (float) Math.PI; 


I 
} 


public float distanceFromPoint(int x1, int y1){ 


return (float) Math.sqrt( 


Math.pow(x1 - x, 2) + 
Math.pow(y1 - y, 2)); 
} 
public Point position() { 
return new Point(x, y); 
} 
// Beasts know how to draw themselves: 
public void draw(Graphics g) { 
g.setColor(color); 
int directionInDegrees = (int)( 
(currentDirection * 360) / (2 * Math.PI)); 
int startAngle = directionInDegrees - 
FieldOBeasts.halfFieldOfView; 
int endAngle = 90; 
g.fillArc(x, y, GSIZE, GSIZE, 


startAngle, endAngle); 


} 


public class FieldOBeasts extends Applet 
implements Runnable { 
private Vector beasts; 
static float 


fieldOfView = 


(float) (Math.PI / 4), // In radians 
// Deceleration % per second: 
decayRate = 1.0f, 
minimumDistance = 10f; // In pixels 
static int 


halfFieldofView 


(int) ( 
(fieldOfView * 360) / (2 * Math.PI)), 
xExtent = 0, 


yExtent 


lI 
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numBeasts = 50, 
maxSpeed = 20; // Pixels/second 
boolean uniqueColors = true; 
Thread thisThread; 
int delay = 25; 
public void init() { 
if (xExtent == © && yExtent == 0) { 
xExtent = Integer.parseInt( 
getParameter("xExtent")); 
yExtent = Integer.parseInt( 
getParameter("yExtent")); 
} 
beasts = 


makeBeastVector(numBeasts, uniqueColors); 


// Now start the beasts a-rovin': 
thisThread = new Thread(this); 
thisThread.start(); 
} 
public void run() { 
while(true) { 
for(int i = 0; i < beasts.size(); i++){ 


Beast b 


(Beast) beasts.elementAt(i); 
b.step(); 
} 


try { 


thisThread.sleep(delay); 
} catch(InterruptedException ex){} 


repaint(); // Otherwise it won't update 


} 


Vector makeBeastVector ( 
int quantity, boolean uniqueColors) { 
Vector newBeasts = new Vector(); 
Random generator = new Random(); 
// Used only if uniqueColors is on: 
double cubeRootOfBeastNumber = 


Math.pow((double)numBeasts, 1.0 / 3.0); 


float colorCubeStepSize = 
(float) (1.0 / cubeRootOfBeastNumber ); 
float r = 0.0f; 


float g 


0.0f; 


float b 


lI 
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.Of; 


for(int i = 0; i < quantity; i++) { 


int x 
(int) (generator.nextFloat() * xExtent); 
if(x > xExtent - Beast.GSIZE) 
x -= Beast.GSIZE; 
int y = 
(int) (generator.nextFloat() * yExtent); 
if(y > yExtent - Beast.GSIZE) 
y -= Beast.GSIZE; 
float direction = (float)( 
generator.nextFloat() * 2 * Math.PI); 
int speed = (int)( 
generator.nextFloat() * (float )maxSpeed); 
if(uniqueColors) { 
r += colorCubeStepSize; 
if(r > 1.0) { 
r -= 1.0f; 


g += colorCubeStepSize; 


if( g > 1.0) { 
g -= 1.0f; 
b += colorCubeStepSize; 
if(b > 1.0) 


b -= 1.0f; 


} 


newBeasts.addElement ( 
new Beast(this, x, y, direction, speed, 
new Color(r,g,b))); 
} 
return newBeasts; 
} 
public Vector beastListInSector(Beast viewer) { 
Vector output = new Vector(); 
Enumeration e = beasts.elements(); 
Beast aBeast = (Beast)beasts.elementAt(0); 
int counter = 0; 
while(e.hasMoreElements()) { 
aBeast = (Beast) e.nextElement(); 
if(aBeast != viewer) { 


Point p = aBeast.position(); 


Point v = viewer.position(); 
float bearing = 
aBeast.bearingFromPointAlongAxis( 
V.X, V.y, viewer.currentDirection); 
if(Math.abs(bearing) < fieldOfView / 2) 


output .addElement(aBeast); 


} 


return output; 

} 

public void paint(Graphics g) { 
Enumeration e = beasts.elements(); 
while(e.hasMoreElements()) { 


((Beast )e.nextElement()).draw(g); 


} 

public static void main(String[] args) { 
FieldOBeasts field = new FieldOBeasts(); 
field.xExtent = 640; 


field.yExtent 


480; 
Frame frame = new Frame("Field 'O Beasts"); 
// Optionally use a command-line argument 


// for the sleep time: 


if(args.length >= 1) 
field.delay = Integer.parseInt(args[0]); 
frame .addwindowListener ( 
new WindowAdapter() { 
public void windowClosing(WindowEvent e) { 
System.exit(0); 
} 
}); 
frame.add(field, BorderLayout.CENTER) ; 
frame.setSize(640, 480); 
field.init(); 
field.start(); 
frame.setVisible(true); 
} 
} ///:~ 


尽管 这 并 非 对 Craig Reynold 的 “Boids” 例 子 中 的 行为 完美 重 现 ， 但 它 却 展 
现 出 了 自己 独 有 的 迷人 之 外 。 通 过 对 数字 进行 调整 ， 即 可 进行 全 面 的 修 
改 。 人 至 于 与 这 种 群 聚 行为 有 关 的 更 多 的 情况 ， 大 家 可 以 访问 Craig 

Sen 在 那个 地 方 ， 甚 至 还 提供 了 Boids 一 个 公开 的 3D 展 示 








http://www.hmt.com/cwr/boids.html 


ee eee 请 在 HTML 文 件 中 设置 下 述 程序 
小 心 : 


<applet 


code=FieldOBeasts 

width=640 

height=480> 

<param name=xExtent value = "640"> 
<param name=yExtent value = "480"> 


</applet> 


17.4 总 结 


通过 本 章 的 学 习 ， 大 家 知道 运用 Java 可 做 到 一 些 较 复杂 的 事情 。 通 过 这 
些 例子 亦 可 看 出 ， 尽 管 Java 必 定 有 自己 的 局 限 ， 但 受 那些 局 限 影响 的 主 
要 是 性 能 (比如 写 好 文字 处 理 程序 后 ， 会 发 现 C++ 的 版 本 要 快 得 多 
这 部 分 是 由 于 IO 库 做 得 不 完善 造成 的 ， 而 在 你 读 到 本 书 的 时 候 ， 情 况 也 
许 已 发 生 了 变化 。 但 Java 的 局 限 也 仅 此 而 已 ， 它 在 语言 表达 方面 的 能 
是 无 以 伦比 的 。 利 用 Java， 几 乎 可 以 表达 出 我 们 想得到 的 任何 事情 。 而 
与 此 同时 ，Java 在 表达 的 方便 性 和 易 读 性 上 ， 也 做 足 了 功夫 。 所 以 在 使 
用 Java 时 ， 一 般 不 会 陷入 其 他 语言 常见 的 那 种 复杂 境地 。 使 用 那些 语言 
时 ， 会 感觉 它们 象 一 个 爱 啼 帅 的 老太婆 ， 哪 有 Java 那 样 清 纯 、 简 练 ! 而 
a 
进一步 的 增强 。 




















17.5 练习 


(1) (稍微 有 些 难度 ) 改写 FieldOBeasts.java， 使 它 的 状态 能 够 保持 固 
定 。 加 上 一 些 按钮 ， 人 允许 用 户 保 存 和 恢复 不 同 的 状态 文件 ， 并 从 它们 断 
掉 的 地 方 开始 继续 运行 。 请 先 参 考 第 10 章 的 CADState.java， 再 决定 具体 
ER F 


JUN 











(2) 《〈 大 作业 ) 以 FieldOBeasts.java 作 为 起 点 ， 构 造 一 个 自动 化 交通 仿真 


ZAIN [e] 


(3) “大 作业 〉 以 ClassScanner.java 作 为 起 点 ， 构 造 一 个 特殊 的 工具 ， 用 
它 找 出 那些 虽然 定义 但 从 未 用 过 的 方法 和 字段 。 


(4) 〈 大 作业 ) 利用 JDBC， 构 造 一 个 联络 管理 程序 。 让 这 个 程序 以 一 个 
平面 文件 数据 库 为 基础 ， 其 中 包含 了 名 字 、 地 址 、 电 话 号 码 、E-mail 地 
址 等 联系 资料 。 应 该 能 同 数 据 库 里 方便 地 加 入 新 名 字 。 刍 入 要 查找 的 名 
字 时 ， 请 采用 在 第 15 章 的 VLookup.java 里 介绍 过 的 那 种 名 字 自 动 填充 技 








如 果 你 不 知道 读 什 么 书 ， 


就 关注 这 个 微 信 号 。 





微 信 公 众 号 名 称 ， 幸福 的 味道 


加 小 编 微 信 一 起 读书 


小 编 微 信号 : 2338856113 


【幸福 的 味道 】 忆 提供 200 个 不 同类 型 的 书 单 





1、 历届 茅盾 文学 奖 获 奖 作品 
2、 每 年 豆 关 ， 当 当 ， 亚 马 逊 年 度 图 书 销售 排行 榜 
3、 25 岁 前 一 定 要 读 的 25 本 书 
4、 有 生 之 年 ， 你 一 定 要 看 的 25 部 外 国 纯 文学 名 著 
5、 有生之年 ， 你 一 定 要 看 的 20 部 中 国 现 当 代 名 著 





6、 美 国 亚 马 进 编辑 推荐 的 一 生 必 读书 单 100 本 
7、 30 个 领域 30 本 不 容错 过 的 入 门 书 

8、 这 20 本 书 ， 是 各 领域 的 襄 峰 之 作 

9、 这 7 本 书 ， 教 你 如 何 高 效 读书 

10、 80 万 书 虫 力 荐 的 “给 五 星 都 不 够 ”的 30 本 书 


关注 “幸福 的 味道 ” 微 信 公众 号 ， 即 可 查看 对 应 书 单 和 得 到 电子 书 


也 可 以 在 我 的 网 站 ( 周 读 ) www.ireadweek.com 这 行 下 载 


附录 A 使 用 非 JAVA 代码 


JAVA 语言 及 其 标准 API《〈 应 用 程序 编程 接口 ) YT Bee iS 
绰 有 余 。 但 在 某 些 情况 下 ， 还 是 必须 使 用 非 JAVA 编码 。 人 例如， 我们 有 

时 要 访问 操作 系统 的 专用 特性 ， 与 特殊 的 硬件 设备 打交道 ， 重 复 使 用 现 
有 的 非 Java 接 口 ， 或 者 要 使 用 “对 时 间 敏 感 ” 的 代码 段 ， 等 等 。 与 非 Java 

代码 的 沟通 要 求 获 得 编译 器 和 "“ 虚 拟 机 ”的 专门 文 持 ， 并 需 附 加 的 工具 将 
Java 代 码 映 射 成 非 Java 代 码 〈 也 有 一 个 简单 方法 : 在 第 15 章 的 “一 个 Web 
应 用 ?小节 中 ， 有 个 例子 解释 了 如 何 利 用 标准 输入 输出 同 非 Java 代 码 连 

fe) 。 目 前 ， 不 同 的 开发 商 为 我 们 提供 了 不 同 的 方案 ， Java 1.1 有 “Java 
HARO” (Java Native Interface，JNI) ， 网 景 提 出 了 自己 的 “Java 运 行 
期 接口 ” (Java Runtime Interface) 计划 ， 而 微软 提供 了 J/Direct、“ 本 源 
HO” (Raw Native Interface, RNI) 以 及 Java/COM 和 集成 方案 。 


各 开发 商 在 这 个 问题 上 所 持 的 不 同 态 度 对 程序 员 是 非常 不 利 的 。 若 Java 
应 用 必须 调用 固有 方法 ， 则 程序 员 或 许 要 实现 固有 方法 的 不 同 版 本 
具体 由 应 用 程序 运行 的 平台 决定 。 程 序 员 也 许 实际 需要 不 同 版 本 的 Java 
代码 ， 以 及 不 同 的 Java 虚 拟 机 。 


奶 一 个 方案 是 CORBA (通用 对 象 请 求 代理 结构 ) ， 这 是 由 OMG (对 象 
管理 组 ， 一 家 非 启 利 性 的 公司 协会 ) 开发 的 一 种 集成 技术 。CORBA 并 
非 任何 语言 的 一 部 分 ， 只 是 实现 通用 通信 息 线 及 服务 的 一 种 规范 。 利 用 
它 可 在 由 不 同 语言 实现 的 对 象 之 间 实 现 “ 相 互 操作 ”的 能 力 。 这 种 通信 息 
线 的 名 字 叫 作 ORB 〈 对 象 请 求 代 理 ) ， 是 由 其 他 开发 商 实现 的 一 种 产 

品 ， 但 并 不 属于 Java 语 言 规范 的 一 部 分 。 


本 附录 将 对 JNI，JDIRECT，RNI，JAVA/COM 集 成 和 CORBA 进 行 概 
述 。 但 不 会 作 更 深层 次 的 探讨 ， 甚 至 有 时 还 假定 读者 已 对 相关 的 概念 和 
技术 有 了 一 定 程度 的 认识 。 但 到 最 后 ， 大 家 应 该 能 够 自行 比较 不 同 的 方 
法 ， 并 根据 自己 要 解决 的 问题 挑选 出 最 恰当 的 一 种 。 


A.1 Java 固 有 接口 
JNI 是 一 种 包容 极 广 的 编程 接口 ， 人 允许 我 们 从 Java 应 用 程序 里 调用 固有 


方法 。 它 是 在 Java 1.1 里 新 增 的 ， 维 持 厦 与 Java 1.0 的 相应 特性 一 一 “固有 
方法 接口 ”(NMI) 一 一 东 种 程度 的 兼容 。NMI 设 计 上 一 些 特点 使 其 未 






































获 所 有 虚拟 机 的 文 持 。 考 夸 到 这 个 原因 ，Java 语 言 将 来 的 版 本 可 能 不 再 
提供 对 NMI 的 支持 ， 这 儿 也 不 准备 讨论 它 。 


目前 ，JNI 只 能 与 用 C 或 C++ 写 成 的 固有 方法 打交道 。 利 用 JNI， 我 们 的 
固有 方法 可 以 ; 


四 创建、 检查 及 更 新 Java 对 象 包括 数组 和 字 串 ) 
调用 Java 方 法 

俘获 和 丢弃 “异常 ” 

= 装载 类 并 获取 类 信息 

进行 运行 期 类 型 检查 


所 以 ， 原 来 在 Java 中 能 对 类 及 对 象 做 的 几乎 所 有 事情 在 固有 方法 中 同样 
可 以 做 到 。 


A.1.1 调用 固有 方法 


我 们 先 从 一 个 简单 的 例子 开始 : 一 个 Java 程 序 调用 固有 方法 ， 后 者 再 调 
用 Win32 的 API 函 数 MessageBox0， 显 示 出 一 个 图 形 化 的 文本 框 。 这 个 例 
子 稍 后 也 会 与 /Direct 一 志 使 用 。 若 您 的 平台 不 是 Win32， 只 需 将 包含 了 
下 述 内 容 的 C 头 : 








#include <windows.h> 
替换 成 : 

#include <stdio.h> 

并 将 对 MessageBox() 的 调用 换 成 调用 printfO) 即 可 。 
第 一 步 是 写 出 对 固有 方法 及 它 的 自 变 量 进行 声明 的 Java 代 码 : 


class ShowMsgBox { 








public static void main(String [] args) { 


ShowMsgBox app = new ShowMsgBox(); 
app.ShowMessage("Generated with JNI"); 
} 
private native void ShowMessage(String msg); 
static { 


System. loadLibrary("MsgImpl"); 





在 固有 方法 声明 的 后 面 ， 跟 随 有 一 个 static 代 码 块 ， 它 会 调用 
System.loadLibrary() 可 在 任何 时 候 调 用 它 ， 但 这 样 做 更 恰当 ) 
System.loadLibrary() 将 一 个 DLL 载 入 内 存 ， 并 建立 同 它 的 链接 。DLL 必 
须 位 于 您 的 系统 路 径 ， 或 者 在 包含 了 Java 类 文件 的 目录 中 。 根 据 具 体 的 
平台 ，JVM 会 自动 添加 适当 的 文件 扩展 名 。 


1. C 头 文件 生成 器 : javah 
现在 编译 您 的 Java 源 文件 ， 并 对 编译 出 来 的 .class 文 件 运 行 javah。javah 


是 [ee 但 由 于 我 们 要 使 用 Java 1.1 JNI， 所 以 必须 指定 -jni 





javah -jni ShowMSgBox 


javah 会 谈 入 类 文件 ， 并 为 每 个 固有 方法 声明 在 C 或 C++ 头 文 件 里 生成 一 
个 函数 原型 。 下 面 是 输出 结果 一 -ShowMsgBox.h 源 文件 〈 为 符合 本 书 
的 要 求 ， 稍 微 进 行 了 一 下 修改 ) : 


/* DO NOT EDIT THIS FILE 











- it is machine generated */ 


#include <jni.h> 


/* Header for class ShowMsgBox */ 
#ifndef _Included_ShowMsgBox 
#define _Included_ShowMsgBox 
#ifdef _ cplusplus 


extern "C" { 


#endif 

/* 
* Class: ShowMsgBox 
* Method: ShowMessage 


* Signature: (Ljava/lang/String; )V 

*/ 

JNIEXPORT void JNICALL 
Java_ShowMsgBox_ShowMessage 

(JNIEnv *, jobject, jstring); 

#ifdef _ cplusplus 

} 
#endif 


#endif 


从 “##fdef_cplusplus” 这 个 预 处 理 引 导 命 an 令 可 以 看 出 ， 该 文件 既 可 由 C 编 
译 右 编译 ， 亦 可 由 C++ 编 译 右 编译 。 第 一 个 检 nclude 命 令 包 括 jni.h 
个 头 文件 ， 作 用 之 一 是 定义 在 文件 其 余部 分 用 到 的 类 型 ，JNIEXPORT 
和 JNICALL 是 一 些 宏 ， 它 们 进行 了 适当 的 扩充 ， 以 便 与 那些 不 同 平台 专 
用 的 引导 命令 配合 JNIEnv，jobject 以 及 jstring 则 是 JNI 数 据 类 型 定义 。 








2. 名 称 管理 和 函数 签名 


JNI 统 一 了 固有 方法 的 命名 规则 ; 这 一 点 是 非常 重要 的 ， 因 为 它 属 于 虚 
拟 机 将 Java 调 用 与 固有 方法 链接 起 来 的 机 制 的 一 部 分 。 从 根本 上 说 ， 所 
有 固有 方法 都 要 以 一 个 "Java” 起 头 ， 后 面 跟随 Java 方 法 的 名 字 ; FRR 
字符 则 作为 分 隔 符 使 用 。 知 Java 回 有 方法 “过 载 ”〈 即 命名 重复 ) ， 那 么 
也 把 函数 签名 退 加 到 名 字 后 面 。 在 原型 前 面 的 注释 里 ， 大 家 可 看 到 回 有 
A 欲 了 解 命名 规则 和 回 有 方法 签名 更 详细 的 情况 ， 请 参考 相应 的 
JNI 文 档 。 





3. 实现 自己 的 DLL 


此 时 ， 我 们 要 做 的 全 部 事情 就 是 写 一 个 C 或 C++ 源 文件 ， 在 其 中 包含 由 
javah 生 成 的 头 文件 ， 并 实现 固有 方法 ， 然 后 编译 它 ， 生 成 一 个 动态 链接 
库 。 这 一 部 分 的 工作 是 与 平台 有 关 的 ， 所 以 我 假定 读者 已 经 知道 如 何 创 
建 一 个 DLL 。 通 过 调用 一 个 win32 API， 下 面 的 代码 实现 了 固有 方法 。 
随后 ， 它 会 编译 和 链接 到 一 个 名 为 MsgImpl.dll 的 文件 里 : 


#include <windows.h> 








#include "ShowMsgBox.h" 
BOOL APIENTRY D1l1lMain( HANDLE hModule, 
DWORD dwReason, void** lpReserved) { 
return TRUE; 
} 
JNIEXPORT void JNICALL 
Java_ShowMsgBox_ShowMessage(JNIEnv * jEnv, 
jobject this, jstring jMsg) { 
const char * msg; 
msg = (*jEnv)->GetStringUTFChars(jEnv, jMsg,0); 


MessageBox(HWND_DESKTOP, msg, 


"Thinking in Java: JNI", 
MB_OK | MB_ICONEXCLAMATION) ; 


(*jEnv)->ReleaseStringUTFChars(jEnv, jMsg,msg); 


若 对 Win32 没 有 兴趣 ， 只 需 跳 过 MessageBox() 调 用 ;最 有 趣 的 部 分 是 它 
周围 的 代码 。 传递 到 固有 方法 内 部 的 自 变 量 是 返回 Java 的 大 门 。 
自 变量 是 类 型 JNIEnv 的 ， 0 Ab i 要 的 所 有 挂钩 (下 一 
再 详细 讲述 ) 。 由 于 方法 的 类 型 不 同 ， 一 个 自 变 量 也 有 自己 不 同 的 仿 
a 对 于 象 上 钢 那 样 的 非 static 方 法 《也 叫 作 实 例 方法 )， 第 二 个 月 变量 
等 价 于 C++ 的 “this” 指 针 ， 并 类 似 于 Java 的 “this”: 都 引用 了 调用 HAN 
法 的 那个 对 象 。 对 于 static 方 法 ， 它 是 对 特定 Class 对 象 的 一 个 引用 ， 方 
法 就 是 在 那个 Class 对 象 里 实现 的 。 


剩余 的 自 变 量 代表 传递 到 固有 方法 调用 里 的 Java 对 象 。 主 类 型 也 是 以 这 
种 形式 传递 的 ， 但 它们 进行 的 “ 按 值 ” 传 递 。 


在 后 面 的 小 节 里 ， 我 们 准备 讲述 如 何 从 一 个 固有 方法 的 内 部 访问 和 控制 
JVM， 同 时 对 上 述 代码 进行 更 详尽 的 解释 。 


A.1.2 访问 JNI 函 数 : JNIEnv 自 变量 


利用 JNI 函 数 ， 程 序 员 可 从 一 个 固有 方法 的 内 部 与 JVM 打 交道 。 正 如 大 
家 在 前 面 的 例子 中 看 到 的 那样 ， BU es el 
自 变量 作为 自己 的 第 一 个 参数 : 间 问 类 型 为 
JNIEnv 的 个 特殊 JNI 数 据 结 构 的 指针 。jNI 数 据 2 5 构 的 一 个 元 素 是 指 
向 由 JVM 生 成 的 一 个 数组 的 指针 ; 该 数组 的 每 个 元 素 都 是 指向 一 个 JNI 
函数 的 指针 。 可 从 固有 方法 的 内 部 发 出 对 JNI 函 数 的 调用 ， 做 法 是 撤消 
对 这 些 指针 的 引用 (具体 的 操作 实际 很 简单 )。 每 种 JVM 都 以 自己 的 方 
IKI Y JNIA ŽI, 但 它们 的 地 址 肯定 位 于 预先 定义 好 的 偏 移 处 。 


利用 JNIEnv 目 变量 ， 程 序 员 可 访问 一 系列 图 数 。 这 些 函 数 可 划分 为 下 述 
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四 进行 类 和 对 象 操作 

国 探 角 对 Java 对 象 的 全 局 和 局 部 引用 

图 访问 实例 字段 和 静态 字段 

四 调用 实例 方法 和 静态 方法 

晶 执 行 字 串 和 数组 操作 

国产 生 和 控制 Java 异 党 

JNI 函 数 的 数量 相当 多 ， 这 里 不 再 详 述 。 相 反 ， 我 会 癌 大 家 揭示 使 用 这 
些 函数 时 背后 的 一 些 基 本 原理 。 欲 了 解 更 详细 的 情况 ， 请 参阅 自己 所 用 
编译 器 的 JNI 文 档 。 

知 观 察 一 下 jnih 头 文件 ， 束 会 发 现在 贞 fdef _cplusplus 预 处 理 器 条 件 的 内 
部 ， 当 由 C++ 编译 器 编译 时 ，JNIEnv 结构 被 定义 成 一 个 类 。 这 个 类 包 
含 了 大 量 内 舰 函 数 。 通 过 一 种 简单 而 且 熟 悉 的 语法 ， 这 些 函 数 让 我 们 可 
以 从 容 访问 JNI 函 数 。 例 如 ， 前 例 包含 了 下 面 这 行 代码 : 
(*jEnv)->ReleaseStringUTFChars(jEnv, jMsg,msg); 


它 在 C++ 里 可 改写 成 下 面 这 个 样子 : 











jEnv->ReleaseStringUTFChars(jMsg,msg); 


大 家 可 注意 到 目 己 不 再 需要 同时 撤消 对 jEnv 的 两 个 引用 ， 相 同 的 指针 不 
再 作为 第 一 个 参数 传递 给 JNI 函 数 调用 。 在 这 些 例 子 剩 下 的 地 方 ， 我 会 
使 用 C++ 风格 的 代码 。 


1. 访问 Java 字 串 


作为 访问 JNI 函 数 的 一 个 例子 ， 请 思考 上 述 的 代码 。 在 这 里 ， 我 们 利用 
JNIEnv 的 自 变 量 jEnv 来 访问 一 个 Java 字 串 。Java 字 串 采 取 的 是 Unicode 格 
式 ， 所 以 假 寿 收 到 这 样 一 个 字 串 ， 并 想 把 它 传 给 一 个 非 Unicode 疯 数 
(如 printf()〉， 首 先 必须 用 JNI 函 数 GetStringUTFChars() 将 其 转换 成 
ASCII 字 符 。 该 函数 能 接收 一 个 Java 字 串 ， 然 后 把 它 转 换 成 UTF-8 字 符 
用 8 位 宽度 容纳 ASCII 值 ， 或 用 16 位 宽度 容纳 Unicode; 大 原始 字 串 的 





内 容 完 全 由 ASCII 构 成 ， 那 么 结果 字 串 也 是 ASCII) 。 


GetStringUTFChars 是 JNIEnv 间 接 指 同 的 那个 结构 里 的 一 个 字段 ， 而 这 个 
字段 又 是 指 同 一 个 函数 的 指针 。 为 访问 JNI 函 数 ， 我 们 用 传统 的 C 语 法 来 
调用 一 个 函数 〈 通 过 指针 ) 。 利 用 上 述 形式 可 实现 对 所 有 JNI 函 数 的 访 
问 。 


A.1.3 传递 和 使 用 Java 对 象 


在 前 例 中 ， 我 们 将 一 个 字 串 传递 给 固有 方法 。 事 实 上 ， 亦 可 将 目 己 创建 
的 Java 对 象 传递 给 固有 方法 。 


在 我 们 的 固有 方法 内 部 ， 可 访问 己 收 到 的 那些 对 象 的 字段 及 方法 。 


为 传递 对 象 ， 声 明 固 有 方法 时 要 采用 原始 的 Java 语 法 。 如 下 例 所 示 ， 
MyJavaClass 有 一 个 public( 公 共 )〉 字段， 以 及 一 个 public 方 法 。 
UseObjects 类 声明 了 一 个 固有 方法 ， 用 于 接收 MyJavaClass 类 的 一 个 对 
象 。 为 调查 固有 方法 是 否 能 控制 自己 的 自 变 量 ， 我 们 设置 了 自 变 量 的 
public 字 段 ， 调 用 固有 方法 ， 然 后 打印 出 public 字 上 段 的 值 。 


class MyJavaClass { 











public void divByTwo() { aValue /= 2; } 
public int aValue; 
} 
public class UseObjects { 
public static void main(String [] args) { 
UseObjects app = new UseObjects(); 
MyJavaClass anObj = new MyJavaClass(); 
anObj.aValue = 2; 
app.changeObject(anObj ); 


System.out.printin("Java: " + anObj.aValue); 


} 


private native void 
changeObject(MyJavaClass obj); 
static { 


System. loadLibrary("UseObjImp1"); 





编译 好 代码 ， 并 将 .class 文 件 传 递 给 javah 后 ， 就 可 以 实现 固有 方法 。 在 
下 面 这 个 例子 中 ， 一 旦 取得 字段 和 方法 ID， 就 会 通过 JNI 函 数 访问 它 
iie 


JNIEXPORT void JNICALL 


Java_Use0bjects_change0bject( 
JNIEnv * env, jobject jThis, jobject obj) { 
jclass cls; 
jfieldID fid; 
jmethodID mid; 
int value; 
cls = env->GetObjectClass(obj); 
fid = env->GetFieldID(cls, 
"aValue", "I"); 
mid = env->GetMethodID(cls, 


"divByTwo", "()V"); 


value = env->GetIntField(obj, fid); 
printf("Native: %d\n", value); 
env->SetIntField(obj, fid, 6); 
env->CallVoidMethod(obj, mid); 
value = env->GetIntField(obj, fid); 


printf("Native: %d\n", value); 





除 第 一 个 目 变 量 外 ，C++ 函 数 会 接收 一 个 jobject， 它 代表 Java 对 象 引 
用 “固有 ”的 那 一 面 一 一 那个 引用 是 我 们 从 Java 代 码 里 传递 的 。 我 们 简单 
地 读 取 aValue， 把 它 打 印 出 来 ， 改 变 这 个 值 ， 调 用 对 象 的 divByTwo0 方 
法 ， 再 将 值 重新 打印 一 遍 。 








为 访问 一 个 字段 或 方法 ， 首 先 必须 获取 它 的 标识 生 。 利 用 适当 的 JNI 函 
数 ， 可 方便 地 取得 类 对 象 、 元 系 名 以 及 签名 信息 。 这 些 函 数 会 返回 一 个 
标识 符 ， 利 用 它 可 访问 对 应 的 元 素 。 尽 管 这 一 方式 显得 有 些 曲折 ， 但 我 
们 的 固有 方法 确实 对 Java 对 象 的 内 部 布局 一 无 所 知 。 因 此 ， 它 必须 通过 
由 JVM 返 回 的 索引 访问 字段 和 方法 。 这 样 一 来 ， 不 同 的 JVM 就 可 实现 不 
同 的 内 部 对 象 布局 ， 同 时 不 会 对 固有 方法 造成 影响 。 











奉 运 行 Java 程 序 ， 就 会 发 现 从 Java 那 一 侧 传 来 的 对 象 是 由 我 们 的 固有 方 
法 处 理 的 。 但 传递 的 到 底 是 什么 呢 ? 是 指针 ， 还 是 Java 引 用 ? 而 且 垃 圾 
收集 器 在 固有 方法 调用 期 间 又 在 做 什么 呢 ? 


二 圾 收集 器 会 在 固有 方法 执行 期 间 持续 运行 ， 但 在 一 次 固有 方法 调用 期 
间 ， 我 们 的 对 象 可 保证 不 会 被 当 作 “ 垃 圾 ”收集 去 。 为 确保 这 一 点 ， 事 先 
创建 了 “局 部 引用 ”， 并 在 固有 方法 调用 之 后 立即 清除 。 由 于 它们 的 “ 生 
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由 于 这 些 引用 会 在 每 次 函数 调用 的 时 候 创建 和 破坏 ， 所 以 不 可 在 static 变 
量 中 制作 固有 方法 的 局 部 副本 《本 地 拷贝 ) 。 知 希望 一 个 引用 在 函数 存 
在 期 间 持续 有 效 ， 束 需要 一 个 全 局 引用 。 全 局 引用 不 是 由 JVM 创 建 的 ， 





但 通过 调用 特定 的 JNI 函 数 ， 程 序 员 可 将 局 部 引用 扩展 为 全 局 引用 。 创 
建 一 个 全 局 引用 时 ， 需 对 引用 对 象 的 “生存 时 间 ” 负 责 。 全 局 引用 《以 及 
它 引 用 的 对 象 ) 会 一 直 留 在 内 存 里 ， 直 到 用 特定 的 JNI 函 数 明确 释放 了 
这 个 引用 。 它 类 似 于 C 的 mallocO 和 free()。 








A.1.4 JNI 和 Java 异 常 

利用 JNI， 可 丢弃 、 捕 捉 、 打 印 以 及 重新 丢弃 Java 异 常 ， 就 象 在 一 个 Java 
程序 里 那样 。 但 对 程序 员 来 说 ， 需 目 行 调用 专用 的 JNI 函 数 ， 以 便 对 异 
各 进行 处 理 。 下 面 列 出 用 于 异 芝 处理 的 一 些 JNI 函 数 : 


aThrow(): 丢弃 一 个 现 有 的 异常 对 象 ， 在 固有 方法 中 用 于 重新 丢弃 一 个 


FF ITs o 
mThrowNew(): 生成 一 个 新 的 异常 对 象 ， 并 将 其 丢弃 。 
mExceptionOccurred(): 判断 一 个 异种 是 否 已 被 丢弃 ， 但 尚未 清除 。 


mExceptionDescribe(): 打印 一 个 异常 和 堆栈 跟踪 信息 。 














mExceptionClear(): 清除 一 个 待 决 的 异常 。 
mFatalError(): 造成 一 个 严重 错误 ， 不 返回 。 


在 所 有 这 些 函 数 中 ， 最 不 能 忽视 的 束 是 ExceptionOccurred() 和 
ExceptionClear()。 大 多 数 JNI 函 数 都 能 产生 异常 ， 而 且 没 有 和 象 在 Java 的 
try 块 内 的 那 种 语言 特性 可 供 利 用 。 所 以 在 每 一 次 JNI 函 数 调 用 之 后 ， 都 
必须 调用 ExceptionOccurred0， 了 解 异 各 是否 已 被 丢弃 。 知 侦 测 到 一 个 
异常 ， 可 选择 对 其 加 以 控制 (可 能 时 还 要 重新 丢弃 它 ) 。 然 和 而， 必须 确 
保 异 常 最 终 被 清除 。 这 可 以 在 自己 的 函数 中 用 ExceptionClear() 来 实现 ; 
若 异 党 被 重新 丢弃 ， 也 可 能 在 其 他 某 些 函 数 中 进行 。 但 无 论 如何 ， 这 一 
工作 是 必 不 可 少 的 。 

我 们 必须 保证 异常 被 彻底 清除 。 人 否则 ， 假 各 在 一 个 异常 待 决 的 情况 下 调 
用 一 个 JNI 函 数 ， 获 得 的 结果 往往 是 无 法 预知 的 。 也 有 少数 几 个 JNI 函 数 
可 在 异常 时 安全 调用 ; 当然， 它们 都 是 专门 的 异常 控制 函数 。 


A.1.5JNI 和 线程 处 理 





由 于 Java 是 一 种 多 线程 语言 ， 几 个 线程 可 能 同时 发 出 对 一 个 固有 方法 的 
调用 《〈 知 另 一 个 线程 发 出 调用 ， 固 有 方法 可 能 在 运行 期 间 和 暂停 ) 。 此 

时 ， 完 全 要 由 程序 员 来 保证 固有 调用 在 多 线程 的 环境 中 安全 进行 。 例 

如 ， 要 防范 用 一 种 未 进行 监视 的 方法 修改 共享 数据 。 此 时 ， 我 们 主要 有 
两 个 选择 : 将 固有 方法 声明 为 “同步 »， 或 在 固有 方法 内 部 采取 其 他 某 些 
策略 ， 确 保 数据 处 理 正 确 地 并 发 进行 。 


此 外 ， 绝 对 不 要 通过 线程 传递 JNIEnv， 因 为 它 指向 的 内 部 结构 是 在 “每 
线程 "的 基础 上 分 配 的 ， 而 且 包 含 了 只 对 那些 特定 的 线程 才 有 意义 的 信 

















A.1.6 使 用 现成 代码 


为 实现 JNI 固 有 方法 ， 最 简单 的 方法 就 是 在 一 个 Java 类 里 编写 固有 方法 
的 原型 ， 编 译 那 个 类 ， 再 通过 javah 运 行 .class 文 件 。 但 假若 我 们 已 有 一 
个 大 型 的 、 早 已 存在 的 代码 库 ， 而 且 想 从 Java 里 调用 它们 ， 此 时 又 该 如 
何 是 好 呢 ? 不 可 将 DLL 中 的 所 有 函数 更 名 ， 使 其 符合 JNI 命 名 规则 ， 这 
种 方案 是 不 可 行 的 。 最 好 的 方法 是 在 原来 的 代码 库 “ 外 面 * 写 一 个 封装 

DLL。Java 代 码 会 调用 新 DLL 里 的 函数 ， 后 者 再 调用 原始 的 DLL 函数 。 

这 个 方法 并 非 仅仅 是 一 种 解决 方案 大 多 数 情 况 下 ， 我 们 甚至 必须 这 样 
做 ， 因 为 必须 面向 对 象 引用 调用 JNI 函 数 ， 和 否则 无 法 使 用 它们 


A.2 微软 的 解雇 方案 


到 本 书 完稿 时 为 止 ， 微 软 仍 未 提供 对 JNI 的 支持 ， 只 是 用 自己 的 专利 方 
法 提供 了 对 非 Java 代 码 调用 的 支持 。 这 一 支持 内 建 到 编译 器 Microsoft 
JVM 以 及 外 部 工具 中 。 只 有 程序 用 Microsoft Java 编 译 器 编译 ， 而 且 只 有 
在 Microsoft Java E MAHL (IVM) 上 运行 的 时 候 ， 本 节 讲 述 的 特性 才 会 有 
效 。 知 计划 在 因特网 上 发 行 目 己 的 应 用 ， 或 者 本 单位 的 内 联网 建立 在 不 
同 平台 的 基础 上 ， 就 可 能 成 为 一 个 严重 的 问题 。 


微软 与 Win32 代 码 的 接口 为 我 们 提供 了 连接 Win32 的 三 种 途 
(1) J/Direct: 方便 调用 Win32 DLL 函数 的 一 种 途径 ， 具 有 某 些 限制 。 


(2) 本 原 接口 (RNI): 可 调用 Win32 DLL 函数 ， 但 必须 自行 解决 “垃圾 
收集 ”问题 。 























(3) Java/COM 集 成 : 可 从 Java 里 直接 揭示 或 调用 COM 服 务 。 
后 续 的 小 节 将 分 别 探讨 这 三 种 技术 。 


写作 本 书 的 时 候 ， 这 些 特 性 均 通 过 了 Microsoft SDK for Java 2.0 beta 2 的 
文 持 。 可 从 微软 公司 的 web 站 点 下 载 这 个 开发 平台 《要 经 历 一 个 痛 兰 的 
选择 过 程 ， 他 们 叫 作 *Active Setup”) > Java SDK 是 一 套 命 令 行 工 具 的 集 
合 ， 但 编译 引擎 可 轻易 嵌入 Developer ”Studio 环 境 ， 以 便 我 们 用 Visual 
J++ 1.1 来 编译 Java 1.1 代 码 。 








A.3 J/Direct 

J/Direct 是 调用 Win32 “DLL 函数 最 简单 的 方式 。 它 的 主要 设计 目标 是 与 
Win32API 打 交道 ， 但 完全 可 用 它 调用 其 他 任何 API。 但 是 ， 上 尽管 这 一 特 
性 非常 方便 ， 但 它 同 时 也 造成 了 某 些 限制 ， 且 降低 了 性 能 (与 RNI 相 
比 ) 。 但 JDirect 也 有 一 些 明显 的 优点 。 首 先 ， 除 希望 调用 的 那个 DLL 里 
的 代码 之 外 ， 没 有 必要 再 编写 额外 的 非 Java 代 码 ， 换 言 之 ， 我 们 不 需要 
一 个 封装 器 或 者 代理 / 存根 DLL。 其 次 ， 函 数 自 变量 与 标准 数据 类 型 之 
间 实 现 了 自动 转换 。 若 必须 传递 用 户 自 定义 的 数据 类 型 ， 那 么 JDirect 可 
能 不 按 我 们 的 希望 工作 。 第 三 ， 就 象 下 例 展 示 的 那样 ， 它 非常 简单 和 直 
接 。 只 需 少 数 几 行 ， 这 个 例子 便 能 调用 Win32 API 函数 MessageBox()， 
它 能 弹出 一 个 小 的 模 态 窗口 ， 并 带 有 一 个 标题 、 一 条 消息 、 一 个 可 选 的 
图 标 以 及 几 个 按钮 。 


public class ShowMsgBox { 


public static void main(String args[]) 
throws UnsatisfiedLinkError { 
MessageBox(0, 
"Created by the MessageBox() Win32 func", 
"Thinking in Java", 0); 
} 


/xx @d11.import("USER32") */ 


private static native int 
MessageBox(int hwndOwner, String text, 


String title, int fuStyle); 





令 人 震惊 的 是 ， 这 里 便 是 我 们 利用 JDirect 调 用 win32 DLL 函数 所 需 的 全 
部 代码 。 其 中 的 关键 是 位 于 示范 代码 底部 的 MessageBox(O 声 明之 前 的 
@dll.import 引 导 命 令 。 它 表面 上 看 是 一 条 注释 ， 但 实际 并 非 如 此 。 它 的 
作用 是 告诉 编译 器 : 引导 命令 下 面 的 函数 是 在 USER32 DLL 里 实现 的 ， 
而 且 应 相应 地 调用 。 我 们 要 做 的 全 部 事情 就 是 提供 与 DLL 内 实现 的 函数 
相符 的 一 个 原型 ， 并 调用 函数 。 但 是 毋 需 在 Java 版 本 里 手工 键入 需要 的 
每 一 个 Win32 API 函 数 ， 一 个 Microsoft Java 包 会 帮 我 们 做 这 件 事 情 〈 很 
TRS EAA) 。 为 了 让 这 个 例子 正常 工作 ， 函 数 必须 “ 按 名 称 ” 由 
DIEL 导 出。 但 是 ， 也 可 以 用 @dllLimport 引 导 命 令 “ 按 顺序 ?链接 。 举 个 例 
子 来 说 ， 我 们 可 指定 函数 在 DLL 里 的 入 口 位 置 。 稍 后 还 会 具体 讲述 
@dll.import 引 导 命 令 的 特性 。 


用 非 Java 代 码 进 行 链接 的 一 个 重要 问题 就 是 函数 参数 的 自动 配置 。 正 如 
大 家 看 到 的 那样 ，MessageBox0 的 Java 声 明 采 用 了 两 个 字 串 自 变 量 ， 但 
原来 的 C 方 案 则 采用 了 两 个 char 指 针 。 编 译 右 会 帮助 我 们 自动 转换 标准 
数据 类 型 ， 同 时 遵照 本 章 后 一 节 要 讲述 的 规则 。 


最 好 ， 大 家 或 许 已 注意 到 了 main() 声 明 中 的 UnsatisfiedLinkError 异 常 。 

在 运行 期 的 时 候 ， 一 旦 链接 程序 不 能 从 非 Java 函 数 里 解析 出 符号 ， 就 会 
触发 这 一 异常 (事件 ) 。 这 可 能 是 由 多 方面 的 原因 造成 的 : .dll 文 件 未 
找到 ; 不 是 一 个 有 效 的 DLL; 或 者 JDirect 未 获 您 所 使 用 的 虚拟 机 的 支 
持 。 为 了 使 DLL 能 被 找到 ， 它 必须 位 于 Windows 或 Windows\System 目 录 
下 ， 位 于 由 PATH 环境 变量 列 出 的 一 个 目录 中 ， 或 者 位 于 和 .class 文 件 相 
同 的 目录 。J/Direct 获 得 了 Microsoft Java 编 译 器 1.02.4213 版 本 及 更 高 版 本 
的 支持 ， 也 获得 了 Microsoft JVM 4.79.2164 及 更 高 版 本 的 支持 。 为 了 解 
自己 编译 器 的 版 本 号 ， 请 在 命令 行 下 运行 JVC， 不 要 加 任何 参数 。 为 了 
人 请 找到 msjava.dll 的 图 标 ， 并 利用 右键 弹出 菜单 观察 它 


A.3.1 @dll.import 引 导 命 令 























作为 使 用 JDirect 唯 一 的 途径 ，@dl.import 引 导 命 令 相当 灵 活 。 它 提供 了 
为 数 众多 的 修改 符 ， 可 用 它们 目 定 义 同 非 Java 代 人 码 建 立 链接 关系 的 方 
式 。 它 亦 可 应 用 于 类 内 的 一 些 方法 ， 或 应 用 于 整个 类 。 也 就 是 说 ， 我 们 
在 那个 类 内 声明 的 所 有 方法 都 是 在 相同 的 DLL 里 实现 的 。 下 和 面 让 我 们 具 
体 研究 一 下 这 些 特性 。 


1. 别名 处 理 和 按 顺 序 链 接 


为 了 使 @dll.import 引 导 命 令 能 象 上 面 显示 的 那样 工作 ，DLL 内 的 函数 必 
须 按 名 字 导 出 。 然 而 ， 我 们 有 时 想 使 用 与 DLL 里 原始 名 字 不 同 的 一 个 名 
F JKA) ， 否 则 函数 就 可 能 按 编号 《比如 按 顺 序 ) 导出， 而 不 是 
按 名 字 导 出 。 下 和 面 这 个 例子 声明 了 FinestraDiMessaggio()( 用 意大利 语 
J 。 正 如 大 家 看 到 的 那样 ， 使 用 的 语法 是 非常 简单 





public class Aliasing { 


public static void main(String args|[]) 
throws UnsatisfiedLinkError { 
FinestraDiMessaggio(0, 
"Created by the MessageBox() Win32 func", 
"Thinking in Java", 0); 
} 
/** Q@dll.import("USER32", 
entrypoint="MessageBox") */ 
private static native int 
FinestraDiMessaggio(int hwndOwner, String text, 


String title, int fuStyle); 


下 面 这 个 例子 展示 了 如 何 同 DLL 里 并 非 按 名 字 导 出 的 一 个 函数 建立 链 
接 ， 那 个 函数 事实 是 按 它 们 在 DLL 里 的 位 置 导 出 的 。 这 个 例子 假设 有 一 
个 名 为 MYMATH 的 DLL， 这 个 DLL 在 位 置 编 号 3 处 包含 了 一 个 函数 。 那 
个 函数 获取 两 个 整数 作为 自 变量 ， 并 返回 两 个 整数 的 和 。 


public class ByOrdinal { 





public static void main(String args[]) 
throws UnsatisfiedLinkError { 
int j=3, k=9; 
System.out.printin("Result of DLL function:" 
+ Add(j,k)); 
} 
/** @dll.import("MYMATH", entrypoint = "#3") */ 
private static native int Add(int op1,int op2); 


} 


可 以 看 出 ， 唯 一 的 蕾 异 束 是 entrypoint 自 变量 的 形式 。 

2. 将 @dll.import 应 用 于 整个 类 

@dll.import 引 导 命 令 可 应 用 于 整个 类 。 也 就 是 说 ， 那 个 类 的 所 有 方法 都 
是 在 相同 的 DLL 里 实现 的 ， 并 具有 相同 的 链接 属性 。 引 导 命 令 不 会 由 子 
类 继承 ， 考 虑 到 这 个 原因 ， 而 且 由 于 DLL 里 的 函数 是 自然 的 static 函 数 ， 
所 以 更 佳 的 设计 方案 是 将 API 函 数 封装 到 一 个 独立 的 类 里 ， 如 下 所 示 : 


/xx @dll.import("USER32") */ 








class MyUser32Access { 


public static native int 


MessageBox(int hwndOwner, String text, 
String title, int fuStyle); 
public native static boolean 
MessageBeep(int uType); 
} 
public class WholeClass { 
public static void main(String args[]) 
throws UnsatisfiedLinkError { 
MyUser32Access.MessageBeep (4); 
MyUser32Access .MessageBox(0, 
"Created by the MessageBox() Win32 func", 


"Thinking in Java", 0); 
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数 ， 所 以 必须 在 调用 它们 时 规定 作用 域 。 大 家 也 许 认 为 必须 用 上 述 的 方 
法 将 所 有 Win32 API Cea. AAAS) 都 映射 成 Java 类 。 但 竹 
运 的 是 ， 根 本 不 必 这 样 做 。 


A.3.2 com.ms.win32 包 


Win32 API 的 体积 相当 庞大 包 侣 了 数 以 生计 的 函数 、 篆 数 以 及 数据 
类 型 。 当 然 ， 我 们 并 不 想 将 每 个 Win32 API 函数 都 写成 对 应 Java 形 式 。 
微软 考虑 到 了 这 个 问题 ， 发 行 了 一 个 Java 包 ， 可 通过 JJDirect 将 Win32 

API 映 射 成 Java 类 。 这 个 包 的 名 字 叫 作 com.ms.win32。 安 装 Java SDK 2.0 
时 ， 知 在 安装 选项 中 进行 了 相应 的 设置 ， 这 个 包 束 会 安装 到 我 们 的 类 路 
径 中 。 这 个 包 由 大 量 Java 类 构成 ， 它 们 完整 再 现 了 Win32 ” API 的 常数 、 

















数据 类 型 以 及 函数 。 包 容 能 力 最 大 的 三 个 类 是 User32.class，Kernel.class 
以 及 Gdi32.class。 它 们 包含 的 是 win32 ”API 的 核心 内 容 。 为 使 用 它们 ， 
只 需 在 自己 的 Java 代 码 里 导入 即 可 。 前 面 的 ShowMsgBox 示 例 可 用 
com.ms.win32 改 写成 下 面 这 个 样子 (这 里 也 考虑 到 了 用 更 恰当 的 方式 使 
用 UnsatisfiedLinkError ) : 


import com.ms.win32.*; 


public class UseWin32Package { 
public static void main(String args[]) { 
try { 
User32.MessageBeep ( 
winm.MB_ICONEXCLAMATION) ; 
User32.MessageBox(0, 
"Created by the MessageBox() Win32 func", 
"Thinking in Java", 
winm.MB_OKCANCEL | 
winm.MB_ICONEXCLAMATION) ; 
} catch(UnsatisfiedLinkError e) { 
System.out.printin("Can’t link Win32 API"); 


System.out.println(e); 





Java 包 是 在 第 一 行 导 入 的 。 现 在 ， 可 在 不 进行 其 他 声明 的 前 提 下 调用 


MessageBeep()#ll MessageBox()ri 2. 7EMessageBeep()#, FRA AB 
包 导 入 时 也 声明 了 Win32 常 数 。 这 些 常数 是 在 大 量 Java 接 口 里 定义 的 ， 
全 部 命名 为 winx (x 代表 欲 使 用 之 常数 的 首 字 母 ，。 


写作 本 书 时 ，com.ms.win32 包 的 开发 仍 示 正式 完成 ， 但 已 可 堪 使 用 。 
A.3.3 汇集 


“汇集 ”(Marshaling〉 是 指 将 一 个 函数 自 变 量 从 它 原始 的 二 进 制 形式 转 

换 成 与 语言 无 关 的 某 种 形式 ， 再 将 这 种 通用 形式 转换 成 适合 调用 函数 采 
用 的 二 进 制 格 式 。 在 前 面 的 例子 中 ， 我 们 调用 了 MessageBox0O 函 数 ， 并 
癌 它 传递 了 两 个 字 串 。MessageBox0O 是 个 C 函 数 ， 而 且 Java 字 串 的 二 进 

制 布局 与 C 字 串 并 不 相同 。 但 尽管 如 此 ， 自 变量 仍 获 得 了 正确 的 传递 。 

这 是 由 于 在 调用 C 代 码 前 ，JJDirect 已 帮 有 我 们 考虑 到 了 将 Java 字 串 转 换 成 
C 字 串 的 问题 。 这 种 情况 适合 所 有 标准 的 Java 类 型 。 下 面 这 张 表 格 总 结 

了 简单 数据 类 型 的 默认 对 应 关系 : 























Java C 

byte BYTEEXCHAR 

short SHORT 或 WORD 

int INT, UINT, LONG，ULONG 或 DWORD 

char TCHAR 

long __int64 

float Float 

double Double 

boolean BOOL 

String LPCTSTR 《只 人 允许 在 OLE 模 式 中 作为 返回 值 ) 


byte[] BYTE * 


Short[] WORD * 
char[] TCHAR * 
int[] DWORD * 


这 个 列表 还 可 继续 下 去 ， 但 已 很 能 说 明 问 题 了 。 大 多 数 情况 下 ， 我 们 不 
必 关 心 与 简单 数据 类 型 之 间 的 转换 问题 。 但 一 旦 必须 传递 用 户 自 定义 类 
型 的 目 变量 ， 情 况 就 立即 变 得 不 同 了 。 例 如 ， 可 能 需要 传递 一 个 结构 化 
的 、 有 用户 目 定义 的 数据 类 型 ， 或 者 需要 把 一 个 指针 传 给 原始 内 存 区 域 。 
在 这 些 情况 下 ， 有 一 些 特殊 的 编译 引导 命令 标记 一 个 Java 类 ， 使 其 能 作 
为 一 个 指针 传 给 结构 (@dll.struct 引 导 命令 ) 。 欲 知 使 用 这 些 关 键 字 的 
细节 ， 请 参考 产品 文档 。 


A.3.4 编写 回调 函数 


有 些 Win32 API 函 数 要 求 将 一 个 函数 指针 作为 自己 的 参数 使 用 。 
Windows API 函 数 随后 就 可 以 调用 目 变量 函数 〈 通 常 是 在 以 后 发 生 特定 
的 事件 时 〉。 这 一 技术 就 叫 作 “回调 函数 ”"。 回 调 函 数 的 例子 包括 窗口 进 
程 以 及 我 们 在 打印 过 程 中 设置 的 回调 (为 后 台 打 印 程序 提供 回调 函数 的 
地 址 ， 使 其 能 更 新 状态 ， 并 在 必要 的 时 候 中 止 打 印 ) 。 


另 一 个 例子 是 API 函 数 EnumWindows0， 它 能 枚 举目 前 系统 内 所 有 顶级 
窗口 。EnumWindows() 要 求 获 取 一 个 函数 指针 作为 自己 的 参数 ， 然 后 搜 
索 由 Windows 内 部 维护 的 一 个 列表 。 对 于 列表 内 的 每 个 窗口 ， 它 都 会 调 
用 回调 函数 ， 将 窗口 句柄 作为 一 个 自 变 量 传 给 回调 。 


为 了 在 Java 里 达到 同样 的 目的 ， 必 须 使 用 com.ms.dll 包 里 的 Callback 类 。 
我 们 从 Callback 里 继承 ， 并 取消 callback()。 这 个 方法 只 能 接近 int 参 数 ， 
并 会 返回 int 或 void。 方 法 签名 和 具体 的 实施 取决 于 使 用 这 个 回调 的 
Windows APIK 2 
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例 ， 并 将 其 作为 函数 指针 传递 给 API 函 数 。 随 后 ，J/Direct 会 帮助 我 们 自 
动 完 成 剩余 的 工作 。 


下 面 这 个 例子 调用 了 Win32 API AEnumWindows(); 
EnumWindowsProc 类 里 的 callback(0) 方 法 会 获取 每 个 顶级 窗口 的 句柄 ， 获 











取 标 题 文字 ， 并 将 其 打印 到 控制 合 窗口 。 


import com.ms.dll.*; 


import com.ms.win32.*; 
class EnumWindowsProc extends Callback { 
public boolean callback(int hwnd, int lparam) { 
StringBuffer text = new StringBuffer(50); 
User32.GetwWindowText ( 
hwnd, text, text.capacity()+1); 
if(text.length() != 0) 
System.out.println(text); 


return true; // to continue enumeration. 


} 


public class ShowCallback { 
public static void main(String args[]) 
throws InterruptedException { 
boolean ok = User32.EnumwWindows ( 
new EnumWindowsProc(), 0); 
if (!ok) 
System.err.printin("EnumWindows failed."); 


Thread.currentThread().sleep(3000); 


对 sleepO 的 调用 允许 窗口 进程 在 main0 退 出 前 完成 。 
A.3.5 其 他 J/Direct 特 性 


通过 @dll.import 引 导 命 令 内 的 修改 从 (标记 〉 ， 还 可 用 到 J/Direct 的 男 两 
项 特性 。 第 一 项 是 对 OLE 函 数 的 简化 访问 ， 第 二 项 是 选择 API 函 数 的 
ANSI 及 Unicode 版 本 。 


根据 约定 ， 所 有 OLE 函 数 都 会 返回 类 型 为 HRESULT 的 一 个 值 ， 它 是 由 

COM 定 义 的 一 个 结构 化 整数 值 。 若 在 COM 那 一 级 编写 程序 ， 并 希望 从 

一 个 OLE 函 数 里 返回 某 些 不 同 的 东西 ， 就 必须 将 一 个 特殊 的 指针 传递 给 
它 一 一 该 指针 指向 函数 即将 在 其 中 填充 数据 的 那个 内 存 区 域 。 但 在 Java 
中 ， 我 们 没有 指针 可 用 ; 此 外 ， 这 种 方法 并 不 简练 。 利 用 J/Direct， 我 们 
可 在 @dll.import 引 导 命 令 里 使 用 ole 修 改 符 ， 从 而 方便 地 调用 OLE 函 数 。 

标记 为 ole 函 数 的 一 个 固有 方法 会 从 Java 形 式 的 方法 签名 (通过 它 决定 返 
回 类 型 ) 自动 转换 成 COM 形 式 的 函数 。 


第 二 项 特性 是 选择 ANSI 或 者 Unicode 字 串 控 制 方法 。 对 字 串 进行 控制 的 
大 多 数 Win32 API 函 数 都 提供 了 两 个 版 本 。 人 例如， 假设 我 们 观察 由 
USER32.DIEL 导 出 的 符号 ， 那 么 不 会 找到 一 个 MessageBox0O 函 数 ， 相 反 
会 看 到 MessageBoxA() 和 MessageBoxW() 函 数 一 一 分 别 是 该 水 数 的 ANSI 
和 Unicode 版 本 。 如 果 在 @dll.import 引 导 命 令 里 不 规定 想 调 用 哪个 版 

本 ，JVM 束 会 试 着 上 自行 判断 。 但 这 一 操作 会 在 程序 执行 时 花费 较 长 的 时 
间 。 所 以 ， 我 们 一 般 可 用 ansi，unicode 或 auto 修 改 符 硬 性 规定 。 


欲 了 解 这 些 特性 更 详细 的 情况 ， 请 参考 微软 公司 提供 的 技术 文档 。 

A.4 本 原 接口 (RNI) 

同 JDirect 相 比 ，RNI 是 一 种 比 非 Java 代 码 复杂 得 多 的 接口 ， 但 它 的 功能 
也 十 分 强大 。RNI 比 JDirect 更 接近 于 JVM， 这 也 使 我 们 能 写 出 更 有 效 的 
代码 ， 能 处 理 固 有 方法 中 的 Java 对 象 ， 而 且 能 实现 与 JVM 内 部 运行 机 制 
更 紧密 的 集成 。 

RNI 在 概念 上 类 似 Sun 公 司 的 JNI。 考 虑 到 这 个 原因 ， 而 且 由 于 该 产品 尚 




















未 正式 完工 ， 所 以 我 只 在 这 里 指出 它们 之 间 的 主要 差异 。 欲 了 解 更 详细 
的 情况 ， 请 参考 微软 公司 的 文档 。 
JNI 和 RNI 之 间 存 在 几 方 面 引 人 注目 的 差异 。 下 面 列 出 的 是 由 msjavah 生 


成 的 C 头 文件 《微软 提供 的 msjavah 在 功能 上 相当 于 Sun 的 javah) ， 应 用 
于 前 面 在 JNI 例 子 里 使 用 的 Java 类 文件 ShowMsgBox。 


/* DO NOT EDIT - 





automatically generated by msjavah */ 
#include <native.h> 
#pragma warning(disable: 4510) 
#pragma warning(disable: 4512) 
#pragma warning(disable: 4610) 
struct Classjava_lang_String; 
#define Hjava_lang_String Classjava_lang_String 
/* Header for class ShowMsgBox */ 
#ifndef _Included_ShowMsgBox 
#define _Included_ShowMsgBox 
#define HShowMsgBox ClassShowMsgBox 
typedef struct ClassShowMsgBox { 
#include <pshpack4.h> 

long MSReserved; 
#include <poppack.h> 
} ClassShowMsgBox; 


#ifdef _ cplusplus 


extern "C" { 

#endif 

__declspec(dllexport) void _ cdecl 

ShowMsgBox_ShowMessage (struct HShowMsgBox *, 
struct Hjava_lang_String *); 

#ifdef _ cplusplus 

} 

#endif 

#endif /* _Included_ShowMsgBox */ 

#pragma warning(default:4510) 

#pragma warning(default:4512) 


#pragma warning(default:4610) 





除 可 读 性 较 兰 外 ， 代 码 里 还 隐藏 着 一 些 技术 性 问题 ， 符 我 一 一 道 来 。 


在 RNI 中 ， 固 有 方法 的 程序 员 知 道 对 象 的 二 进 制 布局 。 这 样 便 人 允许 我 们 
直接 访问 自己 希望 的 信息 ; 我 们 不 必 象 在 JNI 里 那样 获得 一 个 字段 或 方 
法 标识 符 。 但 由 于 并 非 所 有 虚拟 机 都 需要 将 相同 的 二 进 制 布局 应 用 于 目 
己 的 对 象 ， 所 以 上 面 的 固有 方法 只 能 在 Microsoft JVM 下 运行 。 


在 JNI 中 ， 通 过 JNIEnv 目 变量 可 访问 大 量 函 数 ， 以 便 同 JVM 打 交道 。 在 
RNI 中 ， 用 于 控制 JVM 运 作 的 函数 变 成 了 可 直接 调用 。 它 们 中 的 某 一 些 
(如 控制 异常 的 那 一 个 类 似 于 它们 的 JNI“ 兄 弟 ”*。 但 大 多 数 RNI 函 数 都 
有 与 JNI 中 不 同 的 名 字 和 用 途 。 


JNI 和 RNI 最 重大 的 一 个 区 别 是 “垃圾 收集 ”的 模型 。 在 JNI 中 ， 垃 圾 收集 
在 固有 方法 执行 期 间 遵守 与 Java 代 人 码 执行 时 相同 的 规则 。 而 在 RNI 中 ， 
要 由 程序 员 在 固有 方法 活动 期 间 自行 负责 “ 拉 圾 收集 器 ?器 的 启动 与 中 
止 。 默 认 情 况 下 ， 垃 圾 收集 器 在 进入 固有 方法 前 处 于 不 活动 状态 ;这 样 
一 来 ， 程 序 员 就 可 假定 准备 使 用 的 对 象 用 不 着 在 那个 时 间 段 内 进行 垃圾 
































收集 。 然 而 一 旦 固有 方法 准备 长 时 间 执 行 ， 程 序 员 就 应 考虑 激活 垃圾 收 
集 器 一 一 通过 调用 GCEnable() 这 个 RNI 隙 数 (GC 是 “Garbage Collector” H 
缩写 ， 即 “ 拉 圾 收集 ?) 。 


也 存在 与 全 局 句柄 特性 类 似 的 机 制 一 一 程序 员 可 利用 可 保证 特定 的 对 象 
在 GC 活动 期 间 不 至 于 被 当 作 “垃圾 * 收 掉 。 概 念 是 类 似 的 ， 但 名 称 有 所 
5 在 RNI 中 ， 人 们 把 它 叫 作 GCFrames。 


A.4.1 RNI 总 结 


RNI 与 Microsoft JVM 紧 密集 成 这 一 事实 既是 它 的 优点 ， 也 是 它 的 缺点 。 

RNI 比 JNI 复 杂 得 多 ， 但 它 也 为 我 们 提供 了 对 JVM 内 部 活动 的 高 度 控 制 ; 

其 中 包括 垃圾 收集 。 此 外 ， 它 显然 针对 速度 进行 了 优化 ， 采 纳 了 C 程 序 

We 但 除了 微软 的 JVM 之 外 ， 它 并 不 适 于 其 
JVM. 














A.5 Java/COM 集 成 


COM 《以 前 称 为 OLE) 代表 微软 公司 的 “组 件 对 象 模型 ”(Component 
Object Model) ， 它 是 所 有 ActiveX 技 术 (包括 ActiveX 控 件 、Automation 
以 及 ActiveX 文 档 ) 的 基础 。 但 COM 还 包含 了 更 多 的 东西 。 它 是 一 种 特 
殊 的 规范 ， 按 照 它 开发 出 来 的 组 件 对 象 可 通过 操作 系统 的 专门 特性 实 
现 “ 相 互 操 作 ”。 在 实际 应 用 中 ， 为 Win32 系 统 开 发 的 所 有 新 软件 都 与 
COM 有 着 一 定 的 关系 操作 系统 通过 COM 对 象 揭示 出 自己 的 一 些 特 
性 。 由 其 他 厂商 开发 的 组 件 也 可 以 建立 在 COM 的 基础 上 ， 我 们 能 创建 
和 注册 自己 的 COM 组 件 。 通 过 这 样 或 那样 的 形式 ， 如 果 我 们 想 编写 
Win32 代 码 ， 那 么 必须 和 COM 打 交道 。 在 这 里 ， 我 们 将 仅仅 重 述 COM 
编程 的 基本 概念 ， 而 且 假定 读者 已 掌握 了 COM 服 务 器 〈 能 为 COM 客 户 
提供 服务 的 任何 COM 对 象 ) 以 及 COM 客 户 〈 能 从 COM 服 务 器 那里 申请 
服务 的 一 个 COM 对 象 ) 的 概念 。 本 节 将 尽 可 能 地 使 叙述 变 得 简单 。 工 
有 具 实际 的 功能 要 强大 得 多 ， 而 且 我 们 可 通过 更 高 级 的 途径 来 使 用 它们 。 
但 这 也 要 求 对 COM 有 着 更 深刻 的 认识 ， 那 已 经 超出 了 本 附录 的 范围 。 

如 果 您 对 这 个 功能 强大 、 但 与 不 同 平 台 有 关 的 特性 感 兴趣 ， 应 该 研究 
COM 和 微软 公司 的 文档 资料 ， 仔 细 了 阅读 有 关 Java/COM 集 成 的 那 部 分 内 
容 。 如 果 想 获得 更 多 的 资料 ， 回 您 推荐 Dale Rogerson 编 著 的 《Inside 
COM》， 该 书 由 Microsoft Press 于 1997 年 出 版 。 


由 于 COM 是 所 有 新 型 Win32 应 用 程序 的 结构 核心 ， 所 以 通过 Java 代 码 使 














用 (或 揭示 ) COM 服 务 的 能 力 就 显得 尤为 重要 。Java/COM 集 成 无 疑 是 
Microsoft Java 编 译 器 以 及 虚拟 机 最 有 趣 的 特性 。Java 和 和 COM 在 它们 的 模 
型 上 是 如 此 相似 ， 所 以 这 个 集成 在 概念 上 是 相当 直观 的 ， 而 且 在 技术 上 
也 能 轻松 实现 无 颖 结合 为 访问 COM， 几 乎 不 需要 编写 任何 特殊 的 
代码 。 大 多 数 技术 细节 都 是 由 编译 器 和 / 或 虚拟 机 控制 的 。 最 终 的 结果 
便 是 Java 程 序 员 可 象 对 符 原 始 Java 对 象 那 样 对 符 COM 对 象 。 而 且 COM 客 
户 可 象 使 用 其 他 COM 服 务 器 那样 使 用 由 Java 实 现 的 COM 服 务 嚣 。 在 这 
里 提醒 大 家 ， 尺 管 我 使 用 的 是 通用 术语 “COM”， 但 根据 扩展 ， 完 全 可 用 
Java 实 现 一 个 ActiveX Automation 服 务 器 ， 亦 可 在 Java 程 序 中 使 用 一 个 
ActiveX 控 件 。 











Java 和 和 COM 最 引 人 注 目的 相似 之 处 就 是 COM 接 口 与 Java 的 “interface” 关 
键 字 的 关系 。 这 是 接近 完美 的 一 种 相符 ， 因 为 : 


COM 对 象 揭示 出 了 接口 (也 只 有 接口 ) 


o 口 本 身 并 不 具备 实施 方案 ; 要 由 揭示 出 接口 的 那个 COM 对 象 负 


COM 接 口 是 对 语义 上 相关 的 一 组 函数 的 说 明 ; 不 会 揭示 出 任何 数据 


COM 类 将 COM 接 口 组 合 到 了 一 起 。Java 类 可 实现 任意 数量 的 Java 接 
口 。 





COM 有 一 个 引用 对 象 柑 型 ， 程 序 员 永远 不 可 能 “拥有 ”一 个 对 象 ， 只 能 
获得 对 对 象 一 个 或 多 个 接口 的 引用 。Java 也 有 一 个 引用 对 象 模型 一 一 对 
一 个 对 象 的 引用 可 “造型 ”成 对 它 的 茶 个 接口 的 引用 。 


COM 对 象 在 内 存 里 的 “生存 时 间 ” 取 决 于 使 用 对 象 的 客户 数量 ， 夺 这 个 
数量 变 成 零 ， 对 象 束 会 将 自己 从 内 存 中 删 去 。 在 Java 中 ， 一 个 对 象 的 生 
存 时 间 也 由 客户 的 数量 决定 。 大 不 再 有 对 那个 对 象 的 引用 ， 对 象 束 会 等 
候 垃 圾 收集 器 的 处 理 。 


Java 与 COM 之 间 这 种 紧密 的 对 应 关系 不 仅 使 Java 程 序 员 可 以 方便 地 访问 
COM 特 性 ， 也 使 Java 成 为 编写 COM 代 码 的 一 种 有 效 语 言 。COM 是 与 语 
言 无 关 的 ， 但 COM 开 发 事实 上 采用 的 语言 是 C++ 和 Visual Basic。 同 Java 
相 比 ，C++ 在 进行 COM 开 发 时 显得 更 加 强大 ， 并 可 生成 更 有 效 的 代码 ， 
R CIRIE o Visual Basic 比 Java 简 单 得 多 ， 但 它 距离 基础 操作 系统 











太 远 了 ， 而 且 它 的 对 象 模型 并 未 实现 与 COM 很 好 的 对 应 (映射 〉 关 
系 。Java 是 两 者 之 间 一 种 很 好 的 折衷 方案 。 


接 下 来 ， 让 我 们 对 COM 开 发 的 一 些 关 键 问 题 进 行 讨论 。 编 写 Java/COM 
客户 和 服务 器 时 ， 这 些 问题 是 首先 需要 和 弄 清楚 的 。 


A.5.1 COM 基 础 


COM 是 一 种 二 进 制 规范 ， 致 力 于 实施 可 相互 操作 的 对 象 。 例 如 ，COM 
认为 一 个 对 象 的 二 进 制 布局 必须 能 够 调用 另 一 个 COM 对 象 里 的 服务 。 

由 于 是 对 二 进 制 布局 的 一 种 描述 ， 所 以 只 要 某 种 语言 能 生成 这 样 的 一 种 
布局 ， 就 可 通过 它 实现 COM 对 象 。 通 常 ， 程 序 员 不 必 关 注 象 这 样 的 一 
些 低级 细节 ， 因 为 编译 器 可 自动 生成 正确 的 布局 。 例 如 ， 假 设 您 的 程序 
是 用 C++ 写 的 ， 那 么 大 多 数 编 译 器 都 能 生成 符合 COM 规 范 的 一 张 虚拟 函 
数 表 格 。 对 那些 不 生成 可 执行 代码 的 语言 ， 比 如 VB 和 Java， 在 运行 期 则 
会 自动 挂 接 到 COM。 


COM 库 也 提供 了 几 个 基本 的 函数 ， 比 如 用 于 创建 对 象 或 查找 系统 中 一 
个 已 注册 COM 类 的 函数 。 


一 个 组 件 对 象 模 型 的 基本 目标 包括 : 
四 让 对 象 调 用 其 他 对 象 里 的 服务 
四 多 许 新 类 型 对 象 〈 或 更 新 对 象 ) 无 颖 插入 环境 


第 一 点 正 是 面向 对 象 程序 设计 要 解决 的 问题 ， 我 们 有 一 个 客户 对 象 ， 它 
能 向 一 个 服务 器 对 象 发 出 请 求 。 在 这 种 情况 下 ，“ 客 户 * 和 “服务 器 "这 两 
个 术语 是 在 常规 意义 上 使 用 的 ， 并 非 指 一 些 特定 的 硬件 配置 。 对 于 任何 
面向 对 象 的 语言 ， 第 一 个 目标 都 是 很 容易 达到 的 一 一 只 要 您 的 代码 是 一 
个 完整 的 代码 块 ， 同 时 实现 了 服务 器 对 象 代码 以 及 客户 对 象 代码 。 若 改 
变 了 客户 和 服务 器 对 象 相互 间 的 沟通 形式 ， 只 需 简单 地 重新 编译 和 链接 
一 遍 即 可 。 重 新 启动 应 用 程序 时 ， 它 就 会 自动 采用 组 件 的 最 新 版 本 。 


但 假 大 应 用 程序 由 一 些 未 在 自己 控制 之 下 的 组 件 对 象 构 成 ， 情 况 残 会 变 
得 近 然 有 有 异 一 一 我 们 不 能 控制 它们 的 源码 ， 而 且 它 们 的 更 新 可 能 完全 独 
立 于 我 们 的 应 用 程序 进行 。 例 如 ， 当 我 们 在 自己 的 程序 里 使 用 由 其 他 广 
商 开 发 的 ActiveX 控 件 时 ， 就 会 面临 这 一 情况 。 控 件 会 安 六 到 我 们 的 系 









































统 里 ， 我 们 的 程序 能 够 (在 运行 期 定位 服务 器 代码 ， 激 活 对 象 ， 同 它 
建立 链接 ， 然 后 使 用 它 。 以 后 ， 我 们 可 安装 控件 的 新 版 本 ， 我 们 的 应 用 
程序 应 该 仍然 能 够 运行 ， 即 使 在 最 糟 的 情况 下 ， 它 也 应 礼貌 地 报告 一 条 
出 错 消息 ， 比 如 "控件 未 找到 "等 等 ;一 般 不 会 莫名 其 妙 地 挂 起 或 死机 。 


在 这 些 情况 下 ， 我 们 的 组 件 是 在 独立 的 可 执行 代码 文件 里 实现 的 : DLL 
或 EXE。 行 服务 器 对 象 在 一 个 独立 的 可 执行 代码 文件 里 实现 ， 就 需要 由 
操作 系统 提供 的 一 个 标准 方法 ， 从 而 激活 这 些 对 象 。 当 然 ， 我 们 并 不 想 
在 上 自己 的 代码 里 使 用 DLL 或 EXE 的 物理 名 称 及 位 置 ， 因 为 这 些 参数 可 能 
经 常 发 生变 化 。 此 时 ， 我 们 想 使 用 的 是 由 操作 系统 维护 的 一 些 标识 符 。 
另外 ， 我 们 的 应 用 程序 需要 对 服务 器 展示 出 来 的 服务 进行 的 一 个 描述 。 
下 面 这 两 个 小 节 将 分 别 讨论 这 两 个 问题 。 


1. GUID 和 注册 表 


COM 采 用 结构 化 的 整数 值 〈 长 度 为 128 位 ) 唯一 性 地 标识 系统 中 注册 的 
COM 项 目 。 这 些 数 字 的 正式 名 称 叫 作 GUID (Globally Unique 
IDentifier， 全 局 唯一 标识 符 ) ， 可 由 特殊 的 工具 和 生成。 此外， 这些 数字 
可 以 保证 在 “任何 空间 和 时 间 ?” 里 独一无二 ， 没 有 重复 。 在 空间 ， 是 由 于 
数字 生成 器 会 读 取 网 卡 的 ID 号 码 ， 在 时 间 ， 是 由 于 同时 会 用 到 系统 的 日 
期 和 时 间 。 可 用 GUID 标 识 COM 类 〈 此 时 叫 作 CLSID ) 或 者 COM 接 口 
(ID) 。 尽 管 名 字 不 同 ， 但 基本 概念 与 二 进 制 结构 都 是 相同 的 。GUID 
亦 可 在 其 他 环境 中 使 用 ， 这 里 不 再 次 述 。 


GUID 以 及 相关 的 信息 都 保存 在 Windows 注 册 表 中 ， 或 者 说 保存 在 “注册 
数据 库 ”(Registration Database) 中。 这 是 一 种 分 级 式 的 数据 库 ， 内 建 
于 操作 系统 中 ， 容 纳 了 与 系统 软 人 硬件 配置 有 关 的 大 量 信息 。 对 于 
COM， 注 册 表 会 跟踪 系统 内 安装 的 组 件 ， 比 如 它们 的 CLSID、 实 现 它们 
的 可 执行 文件 的 名 字 及 位 置 以 及 其 他 大 量 细 节 。 其 中 一 个 比较 重要 的 细 
节 是 组 件 的 ProgID; ProgID 在 概念 上 类 似 于 GUID， 因 为 它们 都 标识 着 
一 个 COM 组 件 。 区 别 在 于 GUID 是 一 个 二 进 制 的 、 通 过 算法 生成 的 值 。 
Ws R ProgID 是 随同 一 个 CLSID 分 配 
No 





























我 们 说 一 个 COM 组 件 已 在 系统 内 注册 ， 最 起 码 的 一 个 条 件 就 是 它 的 
CLSID 和 它 的 执行 文件 已 存在 于 注册 表 中 (ProgID 通 常 也 已 束 位 )。 在 
后 面 的 例子 里 ， 我 们 主要 任务 吏 是 注册 与 使 用 COM 组 件 。 


注册 表 的 一 项 重要 特点 就 是 它 作为 客户 和 服务 需 对 象 之 间 的 一 个 去 耦 层 
使 用 。 利 用 注册 表 内 保存 的 一 些 信息 ， 客 户 会 激活 服务 器 :其 中 一 项 信 
恩 是 服务 器 执行 模块 的 物理 位 置 。 硅 这 个 位 置 及 生 了 变动 ， 注 册 表 内 的 
信息 就 会 相应 地 更 新 。 但 这 个 更 新 过 程 对 于 客户 来 说 是 “透明 ?或 者 看 不 
见 的 。 后 者 只 需 直 接 使 用 ProgID 或 CLSID 即 可 。 换 名 话说， 注册 表 使 服 
务 器 代码 的 位 置 透明 成 为 了 可 能 。 随 着 DCOM 《〈 分 布 式 COM) 的 引 
入 ， 在 本 地 机 器 上 运行 的 一 个 服务 器 甚至 可 移 到 网 络 中 的 一 台 远 程 机 
a aaa a 
F 


2. 类 型 库 


由 于 COM 具 有 动态 链接 的 能 力 ， 同 时 由 于 客户 和 服务 器 代码 可 以 分 开 
独立 发 展 ， 所 以 客户 随时 都 要 动态 侦 测 由 服务 器 展示 出 来 的 服务 。 这 些 
服务 是 用 “类 型 库 ”(Type Library〉 中 一 种 二 进 制 的 、 与 语言 无 关 的 形式 
描述 的 (就 象 接 口 和 方法 签名 ) 。 它 既 可 以 是 一 个 独立 的 文件 (通常 采 
用 .TLB 扩 展 名 ) ， 也 可 以 是 链接 到 执行 程序 内 部 的 一 种 Win32 资 源 。 运 
行 期 间 ， 客 户 会 利用 类 型 库 的 信息 调用 服务 器 中 的 函数 。 


我 们 可 以 写 一 个 Microsoft Interface Definition Language《〈 微 软 接口 定义 
语言 ，MIDL) 源 文 件 ， 用 MIDL 编 译 器 编译 它 ， 从 而 生成 一 个 .TLB 文 
件 。MIDL 语 言 的 作用 是 对 COM 类 、 接 口 以 及 方法 进行 描述 。 它 在 名 
称 、 语 法 以 及 用 途上 都 类 似 OMB/MCORBA IDL。 然 而 ，Java 程 序 员 不 必 
使 用 MIDL 。 后 面 还 会 讲 到 另 一 种 不 同 的 Microsoft 工 具 ， 它 能 读 入 Java 
类 文件 ， 并 能 生成 一 个 类 型 库 。 


3. COM:HRESULT 中 的 函数 返回 代码 


由 服务 器 展示 出 来 的 COM 函 数 会 返回 一 个 值 ， 采 用 预先 定义 好 的 
HRESULT 类 型 。HRESULT 代 表 一 个 包含 了 三 个 字段 的 整数 。 这 样 便 可 
使 用 多 个 失败 和 成 功 代 码 ， 同 时 还 可 以 使 用 其 他 信息 。 由 于 COM 函 数 
返回 的 是 一 个 HRESULT， 上 所 以 不 能 用 返回 值 从 函数 调用 里 取 回 原始 数 
据 。 知 必须 返回 数据 ， 可 传递 指 同 一 个 内 存 区 域 的 指针 ， 函 数 将 在 那个 
区 域 里 填充 数据 。 我 们 把 这 称 为 “外 部 参数 ”"。 作 为 Java/COM 程 序 员 ， 
我 们 不 必 过 于 关注 这 个 问题 ， 因 为 虚拟 机 会 帮助 我 们 自动 照管 一 切 。 这 
个 问题 将 在 后 续 的 小 节 里 讲述 。 


A.5.2 MS Java/COM 和 集成 























同 C++/COM 程 序 员 相 比 ，Microsoft ” Java 编译 器 、 虚 拟 机 以 及 各 式 各 样 
的 工具 极 大 简化 了 Java/COM 程 序 员 的 工作 。 编 译 器 有 特殊 的 引导 命令 
和 包 ， 可 将 Java 类 当 作 COM 类 对 待 。 但 在 大 多 数 情况 下 ， 我 们 只 需 依 赖 
Microsoft JVM 为 COM 提 供 的 文 持 ， 同 时 利用 两 个 有 力 的 外 部 工具 。 


Microsoft Java Virtual Machine (JVM) 在 COM 和 Java 对 象 之 间 扮 演 了 一 
座 桥梁 的 角色 。 知 将 Java 对 象 创建 成 一 个 COM 服 务 器 ， 那 么 我 们 的 对 象 
仍然 会 在 JVM 内 部 运行 。Microsoft JVM 是 作为 一 个 DLL 实现 的 ， 它 向 操 
作 系 统 展示 出 了 COM 接 口 。 在 内 部 ，JVM 将 对 这 些 COM 接 口 的 函数 调 
用 映射 成 Java 对 象 中 的 方法 调用 。 当 然 ，JVM 必 须知 道 哪 个 Java 类 文件 
对 应 于 服务 器 执行 模块 ; 之 所 以 能 够 找 出 这 方面 的 信息 ， 是 由 于 我 们 事 
前 已 用 Javareg 在 Windows 注 册 表 内 注册 了 类 文件 。Javareg 是 与 Microsoft 
Java SDK 配 套 提 供 的 一 个 工具 程序 ， 能 读 入 一 个 Java 类 文件 ， 生 成 相应 
的 类 型 库 以 及 一 个 GUID， 并 可 将 类 注册 到 系统 内 。 亦 可 用 Javareg 注 册 
远程 服务 器 。 例 如 ， 可 用 它 注 册 在 不 同 机 右上 运行 的 一 个 服务 器 。 


如 果 想 写 一 个 Java/COM 客 户 ， 必 须 经 历 一 系列 不 同 的 步 又 。 

Java/COM"“ 客 户 ” 是 一 些 特殊 的 Java 人 代码， 它们 想 激活 和 使 用 系统 内 注册 
的 一 个 COM 服 务 器 。 同 样 地 ， 虚 拟 机 会 与 COM 服 务 器 沟通 ， 并 将 它 提 
供 的 服务 作为 Java 类 内 的 各 种 方法 展示 揭示) 出来。 男 一 个 Microsoft 
工具 是 jactivex， 它 能 读 取 一 个 类 型 库 ， 并 生成 相应 的 Java 源 文件 ， 在 其 
中 包含 特殊 的 编译 器 引导 命令 。 生 成 的 源 文 件 属于 我 们 在 指定 类 型 库 之 
pe De Nee oe 下 一 步 是 在 自己 的 COM 客 户 Java 源 文件 中 导 

那个 包 。 


接 下 来 让 我 们 讨论 两 个 例子 。 
A.5.3 用 Java 设 计 COM 服 务 器 


本 节 将 介绍 ActiveX 控 件 、Automation 服 务 器 或 者 其 他 任何 符合 COM 规 
范 的 服务 器 的 开发 过 程 。 下 面 这 个 例子 实现 了 一 个 简单 的 Automation 服 
务 器 ， 它 能 执行 整数 加 法 。 我 们 用 setAddend() 方 法 设置 addend 的 值 。 
次 调用 sum() 方 法 的 时 候 ，addend 就 会 添加 到 当前 result 里 。 我 们 用 
getResult() 获 得 result 值 ， 并 用 clear() 重 新 设置 值 。 用 于 实现 这 一 行为 的 
Java 类 是 非常 简单 的 : 


public class Adder { 











private int addend; 
private int result; 
public void setAddend(int a) { addend = a; } 
public int getAddend() { return addend; } 
public int getResult() { return result; } 
public void sum() { result += addend; } 
public void clear() { 

result = 0; 


addend = 0; 


为 了 将 这 个 Java 类 作为 一 个 COM 对 象 使 用 ， 我 们 将 Javareg 工 具 应 用 于 编 
译 好 的 Adder.class 文 件 。 这 个 工具 提供 了 一 系列 选项 ; 在 这 种 情况 下 ， 
我 们 指定 Java 类 文件 名 ("Adder") ， 想 为 这 个 服务 器 在 注册 表 里 置 入 的 
ProgID ("JavaAdder.Adder.1") ， 以 及 想 为 即将 生成 的 类 型 库 指 定 的 名 
= C"JavaAdder.tlb") 。 由 于 尚未 给 出 CLSID， 所 以 Javareg 会 自动 生成 
一 个 。 辱 我 们 再 次 对 同样 的 服务 器 调用 Javareg， 束 会 直接 使 用 现成 的 
CLSID. 


javareg /register 

/class: Adder /progid:JavaAdder.Adder.1 

/typelib:JavaAdder.tlb 

Javareg 也 会 将 新 服务 器 注册 到 Windows 注 册 表 。 此 时 ， 我 们 必须 记 住 将 
Adder.class 复 制 到 WindowsWavavtrustlib 目 录 。 考 虑 到 安全 方面 的 原因 


特别 是 涉及 程序 片 调用 COM 服 务 的 问题 )》， 只 有 在 COM 服 务 器 已 安 
装 到 trustlib 目 录 的 前 提 下 ， 这 些 服务 器 才 会 被 激活 。 


现在 ， 我 们 已 在 上 自己 的 系统 中 安装 了 一 个 新 的 Automation 服 务 器 。 为 进 
行 测 试 ， 我 们 需要 一 个 Automation 控 制 器 ， 而 Automation 控 制 器 就 是 
Visual Basic (VB) 。 在 下 面 ， 大 家 会 看 到 几 行 VB 人 代码。 按照 YB 的 格 
了 RUER f PAE, 用 它 从 用 户 那 里 接收 要 相 加 的 值 。 并 用 一 个 
标签 显示 结果 ， 用 两 个 下 推 按钮 分 别 调用 sum0 和 clear0 方 法 。 最 开始 ， 
我 们 声明 了 个 名 为 Adder 的 对 象 变量 。 在 Form_Load 子 例 程 中 (在 窗 
体 首次 显示 时 载 入 ) ， 会 调用 Adder 自 动 服务 器 的 一 个 新 实例 ， 并 对 窗 
体 的 文本 字段 进行 初始 化 。 一 旦 用 户 按 下 “Sum” 或 者 “Clear” 按 钮 ， 束 会 
调用 服务 器 中 对 应 的 方法 。 


Dim Adder As Object 





Private Sub Form_Load() 
Set Adder = CreateObject("JavaAdder .Adder.1") 
Addend.Text = Adder .getAddend 
Result.Caption = Adder.getResult 
End Sub 
Private Sub SumBtn_Click() 
Adder.setAddend (Addend ,Text ) 
Adder .Sum 
Result.Caption = Adder.getResult 
End Sub 
Private Sub ClearBtn_Click() 
Adder .Clear 
Addend.Text = Adder.getAddend 
Result.Caption = Adder.getResult 


End Sub 


注意 ， 这 上 段 代 码 根本 不 知道 服务 器 是 用 Java 实 现 的 。 


运行 这 个 程序 并 调用 了 CreateObjectO 函 数 以 后 ， 就 会 在 Windows 注 册 表 
里 搜索 指定 的 ProgID。 在 与 ProgID 有 关 的 信息 中 ， 最 重要 的 是 Java 类 文 
件 的 名 字 。 作 为 一 个 啊 应 ， 会 启动 Java 虚 拟 机 ， 而 且 在 JVM 内 部 调用 
Java 对 象 的 实例 。 从 那个 时 候 开 始 ，JVM 就 会 自动 接管 客户 和 服务 器 代 
人 码 之 间 的 交流 。 


A.5.4 用 Java 设 计 COM 客 户 


现在 ， 让 我 们 转 到 另 一 侧 ， 并 用 Java 开 发 一 个 COM 客 户 。 这 个 程序 会 调 
用 系统 已 安装 的 COM 服 务 器 内 的 服务 。 就 目前 这 个 例子 来 说 ， 我 们 使 
用 的 是 在 前 一 个 例子 里 为 服务 器 实现 的 一 个 客户 。 尽 管 代码 在 Java 程 序 
员 的 眼中 看 起 来 比较 熟悉 ， 但 在 幕后 发 生 的 一 切 却 并 不 寻常 。 本 例 使 用 
了 用 Java 写 成 的 一 个 服务 器 ， 但 它 可 应 用 于 系统 内 安装 的 任何 ActiveX 控 
件 、ActiveX Automation 服 务 器 或 者 ActiveX 组 件 只 要 我 们 有 一 个 类 
型 库 。 


首先 ， 我 们 将 Jactivex 工 上 共 应 用 于 服务 器 的 类 型 库 。Jactivex 有 一 系列 选 
项 和 开关 可 供 选 择 。 但 它 最 基本 的 形式 是 读 取 一 个 类 型 库 ， 并 生成 Java 
源 文件 。 这 个 源 文件 保存 于 我 们 的 windowsjava/trustlib 目 录 中 。 通 过 下 
面 这 行 代 码 ， 它 应 用 于 为 外 部 COM Automation 服 务 器 生成 的 类 型 库 : 











jactivex /javatlb JavaAdder.tlb 


Jactivex 完 成 以 后 ， 我 们 再 来 看 看 自己 的 windowsjavay/trustlib 目录 。 此 时 
可 在 其 中 看 到 一 个 新 的 子 目 录 ， 名 为 javaadder。 这 个 目录 包含 了 用 于 新 
包 的 源 文件 。 这 是 在 Java 里 与 类 型 库 的 功能 差不多 的 一 个 库 。 这 些 文件 
需要 使 用 Microsoft 编 译 器 的 专用 引导 命令 : @com。jactivex 生 成 多 个 文 
件 的 原因 是 COM 使 用 多 个 实体 来 描述 一 个 COM 服 务 嚣 ( 男 一 个 原因 是 

我 没有 对 MIDL 文 件 和 Java/COM 工 具 的 使 用 进行 细致 的 调整 ) 。 


名 为 Adder.java 的 文件 等 价 于 MIDEL 文 件 中 的 一 个 coclass 引 导 命 令 : 它 是 
对 一 个 COM 类 的 声明 。 其 他 文件 则 是 由 服务 器 揭示 出 来 的 COM 接 口 的 
Java 等 价 物 。 这 些 接口 〈 比 如 Adder_DispatchDefault,java) 都 属于 “ 遗 
送 ”(Dispatch) 接口 ， 属 于 Automation 控 制 器 与 Automation 服 务 器 之 间 
的 沟通 机 制 的 一 部 分 。Java/COM 和 集成 特性 也 支持 双 接 口 的 实现 与 使 
用 。 但 是 ，IDispatch 和 双 接 口 的 问题 已 超出 了 本 附录 的 范围 。 








在 下 面 ， 大 家 可 看 到 对 应 的 客户 代码 。 第 一 行 只 是 导入 由 jactivex 生 成 的 
包 。 人 然后 创建 并 使 用 COM Automation 服 务 器 的 一 个 实例 ， 就 象 它 是 一 
个 原始 的 Java 类 那样 。 请 注意 行内 的 类 型 模型 ， 其 中 “ 例 示 ”] COM 对象 
( 即 生成 并 调用 它 的 一 个 实例 ) 。 这 与 COM 对 象 模型 是 一 致 的 。 在 
COM 中 ， 程 序 员 永远 不 会 得 到 对 整个 对 象 的 一 个 引用 。 相 反 ， 他 们 只 
能 拥有 对 类 内 实现 的 一 个 或 多 个 接口 的 引用 。 


“ 例 示 ”Adder 类 的 一 个 Java 对 象 以 后 ， 就 相当 于 指示 COM 激 活 服 务 器 ， 

并 创建 这 个 COM 对 象 的 一 个 实例 。 但 我 们 随后 必须 指定 自己 想 使 用 哪 

个 接口 ， 在 由 服务 器 实现 的 接口 中 挑选 一 个 。 这 正 是 类 型 模型 完成 的 工 

作 。 这 儿 使 用 的 是 “默认 遗 送 ”接口 ， 已 古 Automation 控 制 右 用 于 同一 个 

Automation 服 务 堪 通信 的 标准 接口 。 谷 了解 这 方面 的 细节 ， 请 参考 由 

电 (Inside COM》。 请 注意 激活 服务 器 并 选择 一 个 COM 接 口 是 
么 容易 ! 


import javaadder.*; 








public class JavaClient { 
public static void main(String [] args) { 
Adder_DispatchDefault iAdder = 
(Adder_DispatchDefault) new Adder(); 

iAdder.setAddend(3); 

iAdder.sum(); 

iAdder.sum(); 

iAdder.sum(); 


System.out.println(iAdder.getResult()); 


现在 ， 我 们 可 以 编译 它 ， 并 开始 运行 程序 。 


1. com.ms.com 包 


com.ms.com 包 为 COM 的 开发 定义 了 数量 众多 的 类 。 它 支持 GUID 的 使 用 
Variant 〈 变 体 ) 和 SafeArray Automation (安全 数组 自动 ) 类 型 一 一 
能 与 ActiveX 控 件 在 一 个 较 深 的 层次 打交道 ， 并 可 控制 COM 寞 常 。 


由 于 篇 幅 有 限 ， 这 里 不 可 能 涉及 所 有 这 些 主题 。 但 我 想 着 重 强 调 一 下 
COM 寞 第 的 问题 。 根 据 规范 ， 儿 乎 所 有 COM 函 数 都 会 返回 一 个 
HRESULT 值 ， 它 告诉 我 们 冰 数 调用 是 否 成 功 ， 以 及 失败 的 原因 。 但 车 
观察 服务 器 和 客户 代码 中 的 Java 方 法 签名 ， 束 会 及 现 没 有 HRESULT。 
相反 ， 我 们 用 函数 返回 值 从 一 些 函 数 那 里 取 回 数据 。“ 虚 拟 机 ”CVM) 
会 将 Java 风 格 的 函数 调用 转换 成 COM 风 格 的 函数 调用 ， 甚 至 包括 返回 参 
数 。 但 假 各 我 们 在 服务 器 里 调用 的 一 个 函数 在 COM 这 一 级 失败 ， 又 会 
在 虚拟 机 里 出 现 什么 事情 呢 ? 在 这 种 情况 下 ，JVM 会 认为 HRESULT 值 
标志 着 一 次 失败 ， 并 会 产生 类 com.ms.com.ComFailException 的 一 个 固有 
Java 异 常 。 这 样 一 来 ， 我 们 就 可 用 Java 异 常 控制 机 制 来 管理 COM 错 误 ， 
而 不 是 检查 函数 的 返回 值 。 


如 欲 深 入 了 解 这 个 包 内 包含 的 类 ， 请 参考 微软 公司 的 产品 文档 。 
A.5.5 ActiveX/Beans 集 成 


Java/COM 集 成 一 个 有 趣 的 结果 就 是 ActiveX/Beans 的 集成 。 也 就 是 说 ， 
Java Bean 可 包含 到 象 VB 或 任何 一 种 Microsoft Office 产 品 那样 的 ActiveX 
容 右 里 。 而 一 个 ActiveX 控 件 可 包含 到 象 Sun BeanBoxix ff M Beans #45 
里 。Microsoft JVM 会 帮助 我 们 考 上 处 到 所 有 的 细节 。 一 个 ActiveX 控 件 仅 
仪 是 一 个 COM 服 务 器 ， 它 展示 了 预先 定义 好 的 、 请 求 的 接口 。Bean 只 
是 一 个 特殊 的 Java 类 ， 它 遵循 特定 的 编程 风格 。 但 在 写作 本 书 的 时 候 ， 
这 一 集成 仍然 不 能 算 作 完美 。 例 如 ， 虚 拟 机 不 能 将 JavaBeans 事 件 映 射 
成 为 COM 事 件 模 型 。 知 希望 从 ActiveX 容 器 内 部 的 一 个 Bean 里 对 事件 加 
以 控制 ，Bean 必 须 通过 低级 技术 拦截 象 鼠 标 行 动 这 类 的 系统 事件 ， 不 能 
采用 标准 的 JavaBeans 委 托 事 件 模型 。 


抛 开 这 个 问题 不 管 ，ActiveX/Beans 集 成 仍然 是 非常 有 趣 的 。 由 于 牵涉 的 
概念 与 工具 与 上 面 讨论 的 完全 相同 ， 所 以 请 参阅 您 的 Microsoft 文 档 ， 了 
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A.5.6 固有 方法 与 程序 片 的 注意 事项 


固有 方法 为 我 们 带 来 了 安全 问题 的 一 些 考 虑 。 若 您 的 Java 代 码 发 出 对 一 
个 固有 方法 的 调用 ， 就 相当 于 将 控制 权 传递 到 了 虚拟 机 “体系 ”的 外 面 。 
固有 方法 拥有 对 操作 系统 的 完全 访问 权限 ! 当然 ， 如 果 由 自己 编写 固有 
方法 ， 这 正 是 我 们 所 希望 的 。 但 这 对 程序 片 来 说 却 是 不 可 接受 的 一 一 至 
少 不 能 默许 这 样 做 。 我 们 不 想 看 到 从 因特网 远程 服务 器 下 载 回来 的 一 个 
程序 片 自由 自在 地 操作 文件 系统 以 及 机 器 的 其 他 敏感 区 域 ， 除 非特 别 允 
许 它 这 样 做 。 为 了 用 JJDirect，RNI 和 COM 集 成 防止 此 类 情况 的 发 生 ， 只 
有 受到 信任 《委托 ) 的 Java 代 码 才 有 权 发 出 对 固有 方法 的 调用 。 根 据 程 
序 片 的 具体 使 用 ， 必 须 满足 不 同 的 条 件 才 可 放行 。 例 如 ， 使 用 JDirect 的 
一 个 程序 片 必须 拥有 数字 化 签名 ， 指 出 自己 受到 完全 信任 。 在 写作 本 书 
的 时 候 ， 并 不 是 所 有 这 些 安全 机 制 都 已 实现 OF Microsoft SDK for 
a beta ”2 版 本 ) 。 所 以 当 新 版 本 出 现 以 后 ， 请 务必 留意 它 的 文档 说 
HH 。 





























A.6 CORBA 


在 大 型 的 分 布 式 应 用 中 ， 我 们 的 茶 些 要 求 并 非 前 面 讲述 的 方法 能 够 满足 
的 。 举 个 例子 来 说 ， 我 们 可 能 想 同 以 前 遗留 下 来 的 数据 仓库 打交道 ， 或 
者 需要 从 一 个 服务 器 对 象 里 获取 服务 ， 无 论 它 的 物理 位 置 在 哪里 。 在 这 
些 情况 下 ， 都 要 求 茶 种 形式 的 “远程 过 程 调用 ”(RPC) ， 而 且 可 能 要 求 
与 语言 无 关 。 此 时 ，CORBA 可 为 我 们 提供 很 大 的 帮助 。 


CORBA 并 非 一 种 语言 特性 ， 而 是 一 种 集成 技术 。 它 代表 着 一 种 具体 的 
规范 ， 各 个 开发 商 通 过 遵守 这 一 规范 ， 可 设计 出 符合 CORBA 标 准 的 集 
成 产品 。CORBA 规 范 是 由 OMG 开 发 出 来 的 。 这 家 非 赢利 性 的 机 构 致 力 
于 定义 一 个 标准 框架 ， 从 而 实现 分 布 式 、 与 语言 无 关 对 象 的 相互 操作 。 


利用 CORBA， 我 们 可 实现 对 Java 对 象 以 及 非 Java 对 象 的 远程 调用 ， 并 可 
与 传统 的 系统 进行 沟通 一 一 采 用 一 种 "位置 透 明 ” 的 形式 。Java 增 添 了 连 
网 支持 ， 是 一 种 优秀 的 “面向 对 象 " 程 序 设计 语言 ， 可 构建 出 图 形 化 和 非 
图 形 化 的 应 用 (程序 ) 。Java 和 OMG 对 象 模 型 存在 着 很 好 的 对 应 关系 ; 
例如 ， 无 论 Java 还 是 CORBA 都 实现 了 “接口 ”的 概念 ， 并 且 都 拥有 一 个 引 
用 (参考 ) 对 象 模型 。 


A.6.1 CORBA 基 础 




















由 OMG 制 订 的 对 象 相互 操作 规范 通 单 称 为 “对象 管 理 体 

系 ”(ObjectManagement Architecture, OMA) 。OMA 定 义 了 两 个 组 
件 :“ 核 心 对 象 模型 ”(Core Object Model) 和 “OMA 参 考 体系 ”(OMA 
Reference Model) 。OMA 参 考 体系 定义 了 一 套 基层 服务 结构 及 机 制 |， 
实现 了 对 象 相互 间 进 行 操作 的 能 力 。OMA 参 考 体系 包括 “对 象 请 求 代 
理 ”(Object Request Broker, ORB) 、“ 对 象 服 务 ”(Object Services, tH 
称 作 CORBAservices) 以 及 一 些 通 用 机 制 。 


ORB 是 对 象 间 相互 请 求 的 一 条 通信 总线 。 进 行 请 求 时 ， 坪 需 关 心 对 方 的 
物理 位 置 在 哪里 。 这 意味 着 在 客户 代码 中 看 起 来 象 一 次 方案 调用 的 过 程 
实际 是 非常 复杂 的 一 次 操作 。 首 先 ， 必 须 存在 与 服务 器 对 象 的 一 条 连接 
途径 。 而 且 为 了 创建 一 个 连接 ，ORB 必 须知 道具 体 实 现 服务 器 的 代码 存 
放 在 哪里 。 建 好 连接 后 ， 必 须 对 方法 目 变 量 进行 “汇集 ”。 例 如 ， 将 它们 
转换 到 一 个 二 进 制 流 里 ， 以 便 通 过 网 络 传送 。 必 须 传 递 的 其 他 信息 包括 
服务 占 的 机 右 名 称 、 服 务 器 进程 以 及 对 那个 进程 内 的 服务 器 对 象 进 行 标 
识 的 信息 等 等 。 最 后 ， 这 些 信息 通过 一 种 低级 线路 协议 传递 ， 信 息 在 服 
务 器 那 一 端 解码 ， 最 后 正式 执行 调用 。ORB 将 所 有 这 些 复杂 的 操作 都 从 
法 一 样 简单 。 


并 没有 硬性 规定 应 如 何 实现 ORB 核 心 ， 但 为 了 在 不 同 开发 商 的 ORB 之 间 
实现 一 种 基本 的 兼容 ，OMG 定 义 了 一 系列 服务 ， 它 们 可 通过 标准 接口 
访问 。 









































1. CORBA 接 口 定 义 语言 (ODL) 


CORBA 是 面向 语言 的 透明 而 设计 的 : 一 个 客户 对 象 可 调用 属于 不 同类 
的 服务 器 对 象 方法 ， 无 论 对 方 是 用 何 种 语言 实现 的 。 当 然 ， 客 户 对 象 事 
先 必须 知道 由 服务 器 对 象 揭示 的 方法 名 称 及 签名 。 这 时 便 要 用 到 IDL。 
CORBA ” IDL 是 一 种 与 语言 无 天 的 设计 方法 ， 可 用 它 指定 数据 类 型 、 属 
性 、 操 作 、 接 口 以 及 更 多 的 东西 。IDL 的 语法 类 似 于 C++ 或 Java 语 法 。 
TD eee 了 三 种 语言 一 些 通 用 概念 ， 并 展示 了 它们 的 对 
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CORBA IDL Java C++ 








模块 (Module) € (Package) 命名 空间 (Namespace) 


接口 〈Interface) 接口 (Interface〉 纯 抽 象 类 (Pure abstract class ) 
方法 (Method) 方法 (Method) 成 员 函 数 (Member function) 


继承 概念 也 获得 了 支持 一 一 就 象 C++ 那 样 ， 同 样 使 用 冒号 运算 符 。 针 对 
需要 由 服务 器 和 客户 实现 和 使 用 的 属性 、 方 法 以 及 接口 ， 程 序 员 要 写 出 
一 个 IDL 描 述 。 随 后 ，IDEL 会 由 一 个 由 广 商 提供 的 IDLAava 编 译 器 进行 编 
译 ， 后 者 会 读 取 IDL 源 码 ， 并 生成 相应 的 Java 代 码 。 


IDL 编 译 器 是 一 个 相当 有 用 的 工具 : 它 不 仅 生 成 与 IDL 等 价 的 Java 源 码 ， 
也 会 生成 用 于 汇集 方法 目 变 量 的 代码 ， 并 可 发 出 远程 调用 。 我 们 将 这 种 
代码 称 为 “ 根 于 ”(Stub and Skeleton) 代码 ， 它 组 织 成 多 个 Java 源 文件 ， 
而 且 通 党 属于 同一 个 Java 包 的 一 部 分 。 


2. 命名 服务 


命名 服务 属于 CORBA 基 本 服务 之 一 。CORBA 对 象 是 通过 一 个 引用 访问 
的 。 尽 管 引 用 信息 用 我 们 的 眼睛 来 看 没什么 意义 ， 但 可 为 引用 分 配 由 程 
序 员 定义 的 字 串 名 。 这 一 操作 叫 作 * 引 用 的 字 串 化 >”。 一 个 叫 作 ”命名 服 
务 ”(Naming Service) 的 OMA 组 件 专门 用 于 执行 “ 字 串 到 对 象 " 以 及 “对 
象 到 字 串 ”转换 及 映射 。 由 于 命名 服务 扮演 了 服务 器 和 客户 都 能 查询 和 
操作 的 一 个 电话 本 的 角色 ， 所 以 它 作 为 一 个 独立 的 进程 运行 。 创 建 “ 对 
象 到 字 串 ”映射 的 过 程 叫 作 “ 绑 定 一 个 对 象 ?， 删 除 映 射 关系 的 过 程 叫 

作 “ 取 消 绑 定 ” 而 让 对 象 引 用 传递 一 个 字 串 的 过 程 叫 作 “ 解 析 名 称 ” 


比如 在 局 动 的 时 候 ， 服 务 器 应 用 可 创建 一 个 服务 器 对 象 ， 将 对 象 同 命名 
服务 绑 定 起 来 ， 然 后 等 候 客户 发 出 请 求 。 客 户 首先 获得 一 个 服务 器 引 
用 ， 解 析出 字 串 名 ， 然 后 通过 引用 发 出 对 服务 器 的 调用 。 


同样 地 , “命名 服务 ?规范 也 属于 CORBA 的 一 部 分 ， 但 实现 它 的 应 用 程 


序 是 由 ORB 广 商 《〈《 开 及 商 ) 提供 的 。 由 于 广 商 不 同 ， 我 们 访问 命名 服务 
的 方式 也 可 能 有 所 区 别 。 


A.6.2 一 个 例子 
这 儿 显 示 的 代码 可 能 并 不 详尽 ， 因 为 不 同 的 ORB 有 不 同 的 方法 来 访问 


CORBA 服 务 ， 所 以 无 论 什么 例子 都 要 取决 于 具体 的 厂商 (下 例 使 用 了 
JavaIDL， 这 是 Sun 公 司 的 一 个 免费 产品 。 它 配套 提供 了 一 个 简化 版 本 的 




















ORB、 一 个 命名 服务 以 及 一 个 “IDL ., Java” 编译 器 ) 。 除 此 之 外 ， 由 于 
Java 仍 处 在 发 展 初 期 ， 所 以 在 不 同 的 Java/CORBA 产 品 里 并 不 是 包含 了 
所 有 CORBA 特 性 。 

我 们 希望 实现 一 个 服务 器 ， 令 其 在 一 些 机 器 上 运行 ， 其 他 机 器 能 向 它 查 
询 正 确 的 时 间 。 我 们 也 希望 实现 一 个 客户 ， 令 其 请 求 正确 的 时 间 。 在 这 
种 情况 下 ， 我 们 让 两 个 程序 都 用 Java 实 现 。 但 在 实际 应 用 中 ， 往 往 分 别 
采用 不 同 的 语言 。 

1. 编写 IDL 源 码 


第 一 步 是 为 提供 的 服务 编写 一 个 IDL 描 述 。 这 通常 是 由 服务 器 程序 员 完 
成 的 。 随 后 ， 程 序 员 就 可 用 任何 语言 实现 服务 器 ， 只 需 那 种 语言 里 存在 
着 一 个 CORBA IDL 编 译 器 。 

IDL 文 件 已 分 发 给 客户 端的 程序 员 ， 并 成 为 两 种 语言 间 的 桥梁 。 

下 面 这 个 例子 展示 了 时 间 服 务 器 的 IDL 描 述 情况 : 


module RemoteTime { 








interface ExactTime { 
string getTime(); 
ti 
ti 
这 是 对 RemoteTime 命 名 空间 内 的 ExactTime 接 口 的 一 个 声明 。 该 接口 由 
单独 一 个 方法 构成 ， 它 以 字 串 格式 返回 当前 时 间 。 
2. 创建 根 干 


第 二 步 是 编译 IDL， 创 建 Java 根 干 代码 。 我 们 将 利用 这 些 代码 实现 客户 
和 服务 器 。 与 JavaIDL 产 品 配套 提供 的 工具 是 idltojava: 








idltojava -fserver -fclient RemoteTime.idl 


其 中 两 个 标记 告诉 idltojava 同 时 为 根 和 干 生 成 代码 。idltojava 会 生成 一 个 
Java 包 ， 它 在 IDL 模 块 、RemoteTime 以 及 生成 的 Java 文 件 置 入 
RemoteTime 子 目录 后 命名 。_ExactTimeImplBase.java 代 表 我 们 用 于 实现 
IRA aE RAP”; 而 _ExactTimeStub.java 将 用 于 客户 。 在 
ExactTime.java 中 ， 用 Java 方 式 表 示 了 IDL 接 口 。 此 外 还 包含 了 用 到 的 其 
他 支持 文件 ， 例 如 用 于 简化 访问 命名 服务 的 文件 。 


3. 实现 服务 器 和 客户 


大 家 在 下 面 看 到 的 是 服务 器 端 使 用 的 代码 。 服 务 器 对 象 是 在 
ExactTimeServer 类 里 实现 的 。RemoteTimeServer 这 个 应 用 的 作用 是 : 创 
建 一 个 服务 器 对 象 ， 通 过 ORB 为 其 注册 ， 指 定 对 象 引用 时 采用 的 名 称 ， 
然后 “安静 ”地 等 候 客户 发 出 请 求 。 


import 


import 
import 
import 
import 


import 


RemoteTime.*; 


org.omg.CosNaming.*; 
org.omg.CosNaming.NamingContextPackage.*; 
org.omg.CORBA.*; 

java.util.*; 


java.text.*; 


// Server object implementation 


class ExactTimeServer extends _ExactTimeImplBasef{ 


public String getTime(){ 


return DateFormat. 


getTimeInstance(DateFormat.FULL). 
format(new Date( 


System.currentTimeMillis())); 


j 


// Remote application implementation 
public class RemoteTimeServer { 
public static void main(String args[]) { 
try { 
// ORB creation and initialization: 
ORB orb = ORB.init(args, null); 
// Create the server object and register it: 
ExactTimeServer timeServerObjRef = 
new ExactTimeServer(); 
orb.connect(timeServerObjRef ); 
// Get the root naming context: 
org.omg.CORBA.Object objRef = 
orb.resolve_initial_references( 
"NameService"); 
NamingContext ncRef = 
NamingContextHelper .narrow(objRef ); 
// Assign a string name to the 
// object reference (binding): 
NameComponent nc = 
new NameComponent("ExactTime", ""); 
NameComponent path[] = {nc}; 


ncRef.rebind(path, timeServerObjRef); 


// Wait for client requests: 
java.lang.Object sync = 

new java.lang.Object(); 
synchronized(sync) { 


sync.wait(); 


} 
catch (Exception e) { 
System.out.printin( 
"Remote Time server error: " + e); 


e.printStackTrace(System.out); 


正如 大 家 看 到 的 那样 ， 服 务 器 对 象 的 实现 是 非常 简单 的 ; 和 它 是 一 个 普通 
的 Java 类 ， 从 IDL 编 译 器 生成 的 “二 ”代码 中 继承 而 来 。 但 在 与 ORB 以 及 
其 他 CORBA 服 务 进 行 联系 的 时 候 ， 情 况 却 变 得 稍微 有 些 复杂 。 


4. 一 些 CORBA 服 务 


这 里 要 简单 介绍 一 下 JavaIDL 相 关 代 码 所 做 的 工作 (注意 暂时 忽略 了 
CORBA 代 码 与 不 同 厂商 有 关 这 一 事实 )。main() 的 第 一 行 代码 用 于 局 动 
ORB。 而 且 理 所 当然 ， 这 正 是 服务 器 对 象 需要 同 它 进行 沟通 的 原因 。 玖 
在 ORB 初 始 化 以 后 ， 紧 接着 束 创 建 了 一 个 服务 器 对 象 。 实 际 上 ， 它 正式 
名 称 应 该 是 “短期 服务 对 象 *，” 从 客户 那里 接收 请 求 ,，“ 生 存 时 间 ” 与 创建 
它 的 进程 是 相同 的 。 创 建 好 短期 服务 对 象 后 ， 就 会 通过 ORB 对 其 进行 注 
册 。 这 音 味 着 ORB 已 知道 它 的 存在 ， 可 将 请 求 转发 给 它 。 

















到 目前 为 止 ， 我们 拥有 的 全 部 东西 就 是 一 个 timeServerObjRef 只 有 

在 当前 服务 器 进程 里 才 有 效 的 一 个 对 象 引 用 。 下 一 步 是 为 这 个 服务 对 象 
分 配 一 个 字 串 形式 的 名 字 。 客 户 会 根据 那个 名 字 寻 找 服务 对 象 。 我 们 通 
过 命名 服务 (Naming Service) 完成 这 一 操作 。 首 先 ， 我 们 需要 对 命名 
服务 的 一 个 对 象 引 用 。 通 过 调用 resolve_initial_references()， 可 获得 对 命 
名 服务 的 字 串 式 对 象 引 用 (在 JavaIDL 中 是 “NameService”) ， 并 将 这 个 
引用 返回 。 这 是 对 采用 narrow0 方 法 的 一 个 特定 NamingContext 引 用 的 模 
型 。 我 们 现在 可 开始 使 用 命名 服务 了 。 


为 了 将 服务 对 象 同一 个 字 串 形式 的 对 象 引 用 绑 定 在 一 起 ， 我 们 首先 创建 
一 个 NameComponent 对 象 ， 用 “ExactTime” 进 行 初始 化 。“ExactTime” 是 
我 们 想 用 于 绑 定 服务 对 象 的 名 称 字 串 。 随 后 使 用 rebind0) 方 法 ， 这 是 受 
限于 对 象 引 用 的 字 串 化 引用 。 我 们 用 rebind0 分 配 一 个 引用 一 一 即使 它 
已 经 存在 。 而 假若 引用 已 经 存在 ， 那 么 bind0) 会 造成 一 个 异常 。 在 
CORBA 中 ， 名 称 由 一 系列 NameContext 构 成 一 一 这 便 是 我 们 为 什么 要 用 
一 个 数组 将 名 称 与 对 象 引 用 绑 定 起 来 的 原因 。 


服务 对 象 最 好 准备 好 由 客户 使 用 。 此 时 ， 服 务 器 进程 会 进入 一 种 等 候 状 
态 。 同 样 地 ， 由 于 它 是 一 种 “短期 服务 ”， 所 以 生存 时 间 要 受 服务 器 进程 
的 限制 。JavaIDL 目 前 尚未 提供 对 “持久 对 象 *”( 只 要 创建 它们 的 进程 保 
持 运 行 状态 ， 对 象 束 会 一 直 存 在 下 去 ) WLF. 


现在 ， 我 们 已 对 服务 器 代码 的 工作 有 了 一 定 的 认识 。 接 下 来 看 看 客户 代 
人 码 : 


import RemoteTime.*; 














import org.omg.CosNaming.*; 
import org.omg.CORBA.”*; 
public class RemoteTimeClient { 
public static void main(String args[]) { 
try { 


// ORB creation and initialization: 


ORB orb = ORB.init(args, null); 

// Get the root naming context: 

org.omg.CORBA.Object objRef = 
orb.resolve_initial_references( 

"NameService"); 

NamingContext ncRef = 
NamingContextHelper.narrow(objRef); 

// Get (resolve) the stringified object 

// reference for the time server: 
NameComponent nc = 


new NameComponent("ExactTime", ""),; 


NameComponent path[ ] {nc}; 


ExactTime timeObjRef 
ExactTimeHelper .narrow( 
ncRef.resolve(path)); 
// Make requests to the server object: 
String exactTime = timeObjRef.getTime(); 
System.out.println(exactTime) ; 
} catch (Exception e) { 
System. out.printin( 
"Remote Time server error: " + e); 


e.printStackTrace(System.out); 


前 几 行 所 做 的 工作 与 它们 在 服务 露 进 程 里 是 一 样 的 ; ORB 获 得 初始 化 ， 
并 解析 出 对 命名 服务 的 一 个 引用 。 


接 下 来 ， 我 们 需要 用 到 服务 对 象 的 一 个 对 象 引 用 ， 所 以 将 字 串 形式 的 对 
象 引 用 直接 传递 给 resolve() 方 法 ， 并 用 narrow() 方 法 将 结果 造型 到 
ExactTime 接 口 引用 里 。 最 后 调用 getTime()。 


5. 激活 名 称 服 务 进程 


现在 ， 我 们 已 分 别 获得 了 一 个 服务 器 和 一 个 客户 应 用 ， 它 们 已 作 好 相互 
间 进 行 沟通 的 准备 。 大 家 知道 两 者 都 需要 利用 命名 服务 绑 定 和 解析 字 串 
形式 的 对 象 引 用 。 在 运行 服务 或 者 客户 之 前 ， 我 们 必须 启动 命名 服务 进 
程 。 在 JavaIDL 中 ， 命 名 服务 属于 一 个 Java 应 用 ， 是 随 产 品 配套 提供 
的 。 但 它 可 能 与 其 他 产品 有 所 不 同 。JavaIDL 命 名 服务 在 JVM 的 一 个 实 
例 里 运行 ， 并 《默认 ) 监视 网 络 端口 900。 


6. 激活 服务 器 与 客户 


现在 ， 我 们 已 准备 好 局 动 服 务 器 和 客户 应 用 (之 所 以 按 这 一 顺序 ， 是 由 
于 服务 器 的 存在 是 “短期 ”的 ) 。 大 各 个 方面 都 设置 无 误 ， 那 么 获得 的 惑 
古 在 客户 控制 人 台 窗 口内 的 一 行 输出 文字 ， 提 醒 我 们 当前 的 时 间 是 多 少 。 
当然 ， 这 一 结果 本 里 并 没有 什么 令 人 兴奋 的 。 但 应 注意 一 个 问题 ， 即 使 
都 处 在 同一 人 台 机 器 上 ， 客 户 和 服务 器 应 用 仍然 运行 于 不 同 的 虚拟 机 内 。 
1 a a 








这 只 是 一 个 简单 的 例子 ， 面 癌 非 网 络 环境 设计 。 但 通常 将 ORB 配 置 
成 < 与 位 置 无 天 ”。 大 服务 器 与 客户 分 别 位 于 不 同 的 机 器 上 ， 那 么 ORB 可 
用 一 个 名 为 “安装 库 ”(Implementation Repository) 的 组 件 解 析出 远程 字 
串 式 引用 。 尺 管 “ 安 装 库 ”属于 CORBA 的 一 部 分 ， 但 它 几 乎 没有 具体 的 
规格 ， 所 以 各 厂商 的 实现 方式 是 不 尽 相 同 的 。 


正如 大 家 看 到 的 那样 ，CORBA 还 有 许多 方面 的 问题 未 在 这 儿 进 行 详细 
讲述 。 但 通过 以 上 的 介绍 ， 应 已 对 其 有 一 个 基本 的 认识 。 厦 想 获 得 








CORBA 更 详细 的 资料 ， 最 传真 的 起 点 英 过 于 OMB Web 站 点 ， 地 址 是 
http://www.omg.org。 这 个 地 方 提 供 了 丰富 的 文档 资料 、 日 页 、 程 序 以 及 
对 其 他 CORBA 资 源 和 产品 的 链接 。 


A.6.3 Java 程 序 片 和 和 CORBA 


Java 程 序 片 可 扮演 一 名 CORBA 客 户 的 角色 。 这 样 一 来 ， 程 序 片 就 可 访 
问 由 CORBA 对 象 揭示 的 远程 信息 和 服务 。 但 程序 片 只 能 同 最 初 下 载 它 
的 那个 服务 器 连接 ， 所 以 程序 片 与 它 沟 通 的 所 有 CORBA 对 象 都 必须 位 
于 那 台 服务 器 上 。 这 与 CORBA 的 宗旨 是 相 迟 的 : 它 许 话 可 以 实现 “位 置 
的 透明 ”， 或 者 “与 位 置 无 关 ”。 


将 Java 程 序 片 作为 CORBA 客 户 使 用 时 ， 也 会 市 来 一 些 安全 方面 的 问 
题 。 如 果 您 在 内 联网 中 ， 一 个 办 法 是 放宽 对 浏览 器 的 安全 限制 。 或 者 设 
置 一 道 防火 墙 ， 以 便 建立 与 外 部 服务 器 安全 连接 。 


针对 这 一 问题 ， 有 些 Java ORB 产 品 专 门 提供 了 自己 的 解决 方案 。 例 如 ， 
有 些 产 品 实现 了 一 种 名 为 “HTTP 通 道 ”(HTTP Tunneling) WIER, 5 
一 些 则 提供 了 特别 的 防火 墙 功 能 。 


作为 放 到 附录 中 的 内 容 ， 所 有 这 些 主题 都 显得 太 复 全 了 。 但 它们 确实 是 


需要 重点 注意 的 问题 。 

















A.6.4 比较 CORBA 与 RMI 


我 们 已 经 知道 ，CORBA 的 一 项 主要 特性 就 是 对 RPC 《远程 过 程 调用 ) 
的 支持 。 利 用 这 一 技术 ， 我 们 的 本 地 对 象 可 调用 位 置 远程 对 象 内 的 方 
法 。 当 然 ， 目 前 已 有 一 项 固有 的 Java 特 性 可 以 做 完全 相同 的 事情 : 
RMI (参考 第 15 章 ) 。 尽 管 RMI 使 Java 对 象 之 间 进 行 RPC 调 用 成 为 可 
能 ， 但 CORBA 能 在 用 任何 语言 编制 的 对 象 之 间 进 行 RPC。 这 显然 是 一 
项 很 大 的 区 别 。 


然而 ， 可 通过 RMI 调 用 远程 、 非 Java 代 码 的 服务 。 我 们 需要 的 全 部 东西 
就 是 位 于 服务 器 那 一 端的 、 某 种 形式 的 封装 Java 对 象 ， 它 将 非 Java 代 
码 “ 包 庄 * 于 其 中 。 封 装 对 象 通过 RMI 同 Java 客 户 建 立 外 部 连接 ， 并 于 内 
部 建立 与 非 Java 代 人 码 的 连接 一 一 采用 前 面 讲 到 的 某 种 技术 ， 如 JNI 或 
J/Direct。 





使 用 这 种 方法 时 ， 要 求 我 们 编写 某 种 类 型 的 “集成 层 一 这 其 实 正 是 
人 
JORB J 。 


A.7 总结 


我 们 在 这 个 附录 讨论 的 都 是 从 一 个 Java 应 用 里 调用 非 Java 代 码 最 基本 的 
技术 。 每 种 技术 都 有 自己 的 优 缺 点 。 但 目前 最 主要 的 问题 是 并 非 所 有 这 
些 特性 都 能 在 所 有 JVM 中 找到 。 因 此 ， 即 使 一 个 Java 程 序 能 调用 位 于 特 
定 平台 上 的 固有 方法 ， 仍 有 可 能 不 适用 于 安装 了 不 同 JVM 的 另 一 种 平 








Sun 公 司 提供 的 JNI 具 有 灵活 、 简 单 (尽管 它 要 求 对 JVM 内 核 进行 大 量 控 
制 ) 、 功 能 强大 以 及 通用 于 大 多 数 JVM 的 优点 。 到 本 书 完稿 时 为 止 ， 微 
软 仍 未 提供 对 JNI 的 支持 ， 而 是 提供 了 自己 的 J/Direct (调用 Win32 DLL 
函数 的 一 种 简便 方法 ) 和 RNI 特别 适合 编写 高 效率 的 代码 ， 但 要 求 对 
JVM 内 核 有 很 深入 的 理解 ) 。 微 软 也 提供 了 自己 的 专利 Java/COM 集 成 
方案 。 这 一 方案 具有 很 强大 的 功能 ， 且 将 Java 变 成 了 编写 COM 服 务 器 和 
客户 的 有 效 语言 。 只 有 微软 公司 的 编译 器 和 JVM 能 提供 对 JJDirect、RNI 
以 及 Java/COM 的 支持 。 


我 们 最 后 研究 的 是 CORBA， 它 使 我 们 的 Java 对 象 可 与 其 他 对 象 沟通 
一 一 无 论 它们 的 物理 位 置 在 哪里 ， 也 无 论 是 用 何 种 语言 实现 的 。 
CORBA 与 前 面 提 到 的 所 有 技术 都 不 同 ， 因 为 它 并 未 集成 到 Java 语 言 
里 ， 而 是 采用 了 其 他 厂商 (第 三 方 ) 的 集成 技术 ， 并 要 求 我 们 购买 其 他 
厂商 提供 的 ORB。CORBA 是 一 种 有 趣 和 通用 的 方案 ， 但 如 果 只 是 想 发 
出 对 操作 系统 的 调用 ， 它 也 许 并 非 一 种 最 佳 方案 。 





附录 也 对 比 C++ 和 Java 


“作为 一 名 C++ 程序 员 ， 我 们 早已 掌握 了 面 同 对 象 程序 设计 的 基本 概 
念 ， 而 且 Java 的 语法 无 疑 是 非常 熟悉 的 。 事 实 上 ，Java 本 来 束 是 从 
C++ 衍生 出 来 的 。” 


然而 ，C++ 和 Java 之 间 仍 存在 一 些 显 蔷 的 差异 。 可 以 这 样 说 ， 这 些 差 异 
代表 着 技术 的 极 大 进步 。 一 旦 我 们 和 弄 清 楚 了 这 些 差 异 ， 束 会 理解 为 什么 
说 Java 是 一 种 优秀 的 程序 设计 语言 。 本 附录 将 引导 大 家 认识 用 于 区 分 
Java 和 C++ 的 一 些 重要 特征 。 


(1) 最 大 的 障碍 在 于 速度 : 解释 过 的 Java 要 比 C 的 执行 速度 慢 上 约 20 倍 。 
无 论 什 么 都 不 能 阻止 Java 语 言 进行 编译 。 写 作 本 书 的 时 候 ， 刚 刚 出 现 了 
一 些 准 实时 编译 器 ， 它 们 能 显著 加 快速 度 。 当 然 ， 我 们 完全 有 理由 认为 
会 出 现 适 用 于 更 多 流行 平台 的 纯 回 有 编译 器 ， 但 假 符 没有 那些 编译 器 ， 
由 于 速度 的 限制 ， 必 须 有 些 问 题 是 Java 不 能 解决 的 。 


(2) 和 C++ 一 样 ，Java 也 提供 了 两 种 类 型 的 注释 。 


(3) 所 有 东西 都 必须 置 入 一 个 类 。 不 存在 全 局 函数 或 者 全 局 数据 。 如 果 
想 获 得 与 全 局 函数 等 价 的 功能 ， 可 考虑 将 static 方 法 和 static 数 据 置 入 一 
个 类 里 。 注 意 没 有 象 结 构 、 枚 举 或 者 联合 这 一 类 的 东西 ， 一 切 只 

有 “类 ” (Class) ! 


(4) ”所 有 方法 都 是 在 类 的 主体 定义 的 。 所 以 用 C++ 的 眼光 看 ， 似 乎 所 有 
函数 都 已 扇 入 ， 但 实情 并 非 如 何 〈 骨 入 的 问题 在 后 面 讲述 ) 。 


(5) 在 Java 中 ， 类 定义 采取 几乎 和 C++ 一 样 的 形式 。 但 没有 标志 结束 的 分 
号 。 没 有 class foo 这 种 形式 的 类 声明 ， 只 有 类 定义 。 




















class aType() 

void aMethod() {/* 方法 主体 */} 

} 

(6) Java 中 没有 作用 域 范围 运算 符 “::”。Java 利 用 点 号 做 所 有 的 事情 ， 但 


可 以 不 用 考虑 它 ， 因 为 只 能 在 一 个 类 里 定义 元 素 。 即 使 那些 方法 定义 ， 
也 必须 在 一 个 类 的 内 部 ， 所 以 根本 没有 必要 指定 作用 域 的 范围 。 我 们 注 
意 到 的 一 项 差异 是 对 static 方 法 的 调用 : 使 用 ClassName.methodName()。 
除 此 以 外 ，package〈 包 ) 的 名 字 是 用 点 号 建立 的 ， 并 能 用 import 关 键 字 
实现 C++ 的 '“#include" 的 一 部 分 功能 。 例 如 下 面 这 个 语句 : 


import java.awt.*; 
(#include 并 不 直接 映射 成 import， 但 在 使 用 时 有 类 似 的 感觉 。) 


(7) 与 C++ 类 似 ，Java 含 有 一 系列 “ 主 类 型 ”(Primitive type) ， 以 实现 更 
有 效率 的 访问 。 在 Java 中 ， 这 些 类 型 包括 boolean，char，byte，Sshort， 
int，long，float 以 及 double。 所 有 主 类 型 的 大 小 都 是 固有 的 ， 且 与 具体 
的 机 器 无 关 〈 考 虑 到 移植 的 问题 )》。 这 肯定 会 对 性 能 造成 一 定 的 影响 ， 
ee ales 对 类 型 的 检查 和 要 求 在 Java 里 变 得 更 苛刻 。 例 
H: 


四 条 件 表 达 式 只 能 是 boolean〈 布 尔 ) 类 型 ， 不 可 使 用 整数 。 


罩 必 须 使 用 象 X+Y 这 样 的 一 个 表达 式 的 结果 ; 不 能 仅仅 用 “X+Y? 来 实 
现 “ 副 作用 ”。 


(8) ” char 字符 〉 类 型 使 用 国际 通用 的 16 位 Unicode 字 人 符 集 ， 所 以 能 目 动 
表达 大 多 数 国 家 的 字符 。 


(9) “静态 引用 的 字 串 会 自动 转换 成 String 对 象 。 和 C 及 C++ 不 同 ， 没 有 独 
立 的 静态 字符 数组 字 串 可 供 使 用 。 


(10) Java 增 添 了 三 个 右 移 位 运算 和 全“>>>”， 具 有 与 “多 辑 ” 右 移 位 运算 符 类 
似 的 功用 ， 可 在 最 末尾 插入 零 值 。“>>” 则 会 在 移 位 的 同时 插入 符号 位 
〈《 即 “算术 ” 移 位 ) 。 


(11) 尽管 表面 上 类 似 ， 但 与 C++ 相 比 ，Java 数 组 采用 的 是 一 个 颇 为 不 同 
的 结构 ， 并 具有 独特 的 行为 。 有 一 个 只 读 的 length 成 员 ， 通 过 它 可 知道 
数组 有 多 大 。 而 且 一 旦 超过 数组 边界 ， 运 行 期 检查 会 自动 丢弃 一 个 异 

常 。 所 有 数组 都 是 在 内 存 “ 堆 "里 创建 的 ， 我 们 可 将 一 个 数组 分 配给 另 一 
个 (只 是 简单 地 复制 数组 句柄 )。 数 组 标识 符 属于 第 一 级 对 象 ， 它 的 所 
有 方法 通常 都 适用 于 其 他 所 有 对 象 。 

















(12) 对 于 所 有 不 属于 主 类 型 的 对 象 ， 都 只 能 通过 new 命 令 创建 。 和 
C++ 不 同 ，Java 没 有 相应 的 命令 可 以 “在 堆栈 上 ?创建 不 属于 主 类 型 的 对 
象 。 所 有 主 类 型 都 只 能 在 堆栈 上 创建 ， 同 时 不 使 用 new 命 令 。 所 有 主要 
的 类 都 有 目 己 的 “封装 《器 ) ”类 ， 所 以 能 够 通过 new 创 建 等 价 的 、 以 内 
存 “ 扒 "为 基础 的 对 象 〈 主 类 型 数组 是 一 个 例外 : 它们 可 象 C++ 那 样 通过 
集合 初始 化 进行 分 配 ， 或 者 使 用 new) 。 


(13) Java 中 不 必 进 行 提 前 声明 。 知 想 在 定义 前 使 用 一 个 类 或 方法 ， 只 需 
直接 使 用 它 即 可 一 一 编译 器 会 保证 使 用 恰当 的 定义 。 所 以 和 在 C++ 中 不 
同 ， 我 们 不 会 碰 到 任何 涉及 提前 引用 的 问题 。 


(14) Java 没有 预 处 理 机 。 寿 想 使 用 另 一 个 库 里 的 类 ， 只 需 使 用 import 命 
令 ， 并 指定 库 名 即 可 。 不 存在 类 似 于 预 处 理 机 的 宏 。 


(15) Java 用 包 代 蔡 了 命名 空间 。 由 于 将 所 有 东西 都 置 入 一 个 类 ， 而 且 由 
于 采用 了 一 种 名 为 “封装 ”的 机 制 ， 它 能 针对 类 名 进行 类 似 于 命名 空间 分 
解 的 操作 ， 所 以 命名 的 问题 不 再 进入 我 们 的 考虑 之 列 。 数 据 包 也 会 在 早 
独 一 个 库 名 下 收集 库 的 组 件 。 我 们 只 需 简 单 地 “import”( 导 入 ) 一 个 
包 ， 剩 下 的 工作 会 由 编译 器 上 自动 完成 。 


(16) 被 定义 成 类 成 员 的 对 象 句柄 会 目 动 初始 化 成 null。 对 基本 类 数据 成 
员 的 初始 化 在 Java 里 得 到 了 可 靠 的 保障 。 大 不 明确 地 进行 初始 化 ， 它 们 
就 会 得 到 一 个 默认 值 〈《 零 或 等 价 的 值 ) 。 可 对 它们 进行 明确 的 初始 化 

〈 显 式 初始 化 ) : 要 么 在 类 内 定义 它们 ， 要 么 在 构建 器 中 定义 。 采 用 的 
语法 比 C++ 的 语法 更 容易 理解 ， 而 且 对 于 static 和 非 static 成 员 来 说 都 是 固 
人 


(17) “在 Java 里 ， 没 有 象 C 和 C++ 那样 的 指针 。 用 new 创 建 一 个 对 象 的 时 
候 ， 会 获得 一 个 引用 〈 本 书 一 直 将 其 称 作 “句柄 ”) 。 例 如 : 

















String s = new String("howdy"); 


然而 ，C++ 引 用 在 创建 时 必须 进行 初始 化 ， 而 且 不 可 重 定 义 到 一 个 不 同 
的 位 置 。 但 Java 引 用 并 不 一 定局 限于 创建 时 的 位 置 。 它 们 可 根据 情况 任 
意 定义 ， 这 便 消除 了 对 指针 的 部 分 需求 。 在 C 和 C++ 里 大 量 及 用 指针 的 
为 一 个 原因 是 为 了 能 指 加 任意 一 个 内 存 位 置 ( 这 同时 会 使 它们 变 得 不 安 
全 ， 也 是 Java 不 提供 这 一 文 持 的 原因 ) 。 指 针 通 钊 被 看 作 在 基本 变量 数 








组 中 四 处 移动 的 一 种 有 效 手 段 。Java 人 允许 我 们 以 更 安全 的 形式 达到 相同 
的 目标 。 解 决 指针 间 题 的 终极 方法 是 “固有 方法 ”( 已 在 附录 A 讨论 ) 。 
将 指针 传递 给 方法 时 ， 通 常 不 会 市 来 太 大 的 问题 ， 因 为 此 时 没有 全 局 函 
数 ， 只 有 类 。 而 且 我 们 可 传递 对 对 象 的 引用 。Java 语 言 最 开始 声称 目 
己 “ 完 全 不 采用 指针 ! ”但 随 大 许多 程序 员 痢 质问 没有 指针 如 何 工作 ? 于 
是 后 来 又 声明 “采用 受到 限制 的 指针 ”。 大 家 可 目 行 判断 它 是 否 “ 真 ”的 是 
一 个 指针 。 但 不 管 在 何 种 情况 下 ， 都 不 存在 指针 “算术 ”。 


(18) Java 提 供 了 与 C++ 类 似 的 “构建 器 ”(〈Constructor) 。 如 果 不 自 己 定义 
一 个 ， 束 会 获得 一 个 默认 构建 器 。 而 如 果 定 义 了 一 个 非 默 认 的 构建 器 ， 

就 不 会 为 我 们 自动 定义 默认 构建 器 。 这 和 C++ 是 一 样 的 。 注 意 没 有 复制 
构建 器 ， 因 为 所 有 目 变 量 都 是 按 引 用 传递 的 。 


(19) Java 中 没有 “破坏 器 ”(Destructor) 。 变 量 不 存在 “作用 域 > 的 问题 。 
一 个 对 象 的 “存在 时 间 ” 是 由 对 象 的 存在 时 间 决 定 的 ， 并 非 由 垃圾 收集 器 
决定 。 有 个 finalize() 方 法 是 每 一 个 类 的 成 员 ， 它 在 某 种 程度 上 类 似 于 

C++ 的 “破坏 器 >”。 但 finalize0 是 由 垃圾 收集 器 调用 的 ， 而 且 只 负责 释 

MOR” OFFF BRF mO, URLE) 。 如 需 在 一 个 特 
定 的 地 点 做 某 样 事情 ， 必 须 创 建 一 个 特殊 的 方法 ， 并 调用 它 ， 不 能 依赖 
finalize0。 而 在 另 一 方面 ，C++ 中 的 所 有 对 象 都 会 〈 或 者 说 “应 该 >) AK 
坏 ， 但 并 非 Java 中 的 所 有 对 象 都 会 被 当 作 “ 拉 圾 ?收集 掉 。 由 于 Java 不 文 
持 破 坏 器 的 概念 ， 所 以 在 必要 的 时 候 ， 必 须 谨慎 地 创建 一 个 清除 方法 。 
而 且 针 对 类 内 的 基础 类 以 及 成 员 对 象 ， 需 要 明确 调用 所 有 清除 方法 。 


(20) Java 具 有 方法 “过 载 ? 机 制 ， 它 的 工作 原理 与 C++ 函数 的 过 载 几乎 是 
完全 相同 的 。 


(21) Java 不 文 持 默认 上 自 变量 。 


(22) ”Java 中 没有 goto。 它 采取 的 无 条 件 跳 转 机 制 是 “break 标签 ”或 
“continue PRÈ”, FAAS Bk SHINS HEA. 


(23) Java 采 用 了 一 种 单 根 式 的 分 级 结构 ， 因 此 所 有 对 象 都 是 从 根 类 
Object 统一 继承 的 。 而 在 C++ 中 ， 我 们 可 在 任何 地 方 司 动 一 个 新 的 继承 
树 ， 所 以 最 后 往往 看 到 包含 了 大 量 树 的 “一 片 森 林 ”。 在 Java 中 ， 我 们 无 
论 如 何 都 只 有 一 个 分 级 结构 。 尺 管 这 表面 上 看 似乎 造成 了 限制 ， 但 由 于 
我 们 知道 每 个 对 象 肯定 至 少 有 一 个 Object 接 口 ， 所 以 往往 能 获得 更 强大 
的 能 力 。C++ 目 前 似乎 是 唯一 没有 强制 单 根 结构 的 唯一 一 种 OO 语言 。 





























(24) Java 没 有 模板 或 者 参数 化 类 型 的 其 他 形式 。 它 提供 了 一 系列 集合 : 
Vector (向量 ) ，Stack CEFR) 以 及 Hashtable〈 散 列表 ) ， 用 于 容纳 
Object 引用 。 利 用 这 些 集合 ， 我 们 的 一 系列 要 求 可 得 到 满足 。 但 这 些 集 
合并 非 是 为 实现 象 Ct+“ 标 准 模 板 库 ”(STL) 那样 的 快速 调用 而 设计 
的 。Java 1.2 中 的 新 集合 显得 更 加 完整 ， 但 仍 不 具备 正宗 模板 那样 的 高 
效率 使 用 手段 。 


(25) “垃圾 收集 ”意味 着 在 Java 中 出 现 内 存 漏洞 的 情况 会 少 得 多 ， 但 也 并 
非 完 全 不 可 能 ( 寿 调 用 一 个 用 于 分 配 存 储 空间 的 固有 方法 ， 垃 圾 收集 器 
就 不 能 对 其 进行 跟踪 监视 )。 然 而 ， 内 存 漏洞 和 资源 漏洞 多 是 由 于 编写 
不 当 的 finalizeO) 造 成 的 ， 或 是 由 于 在 已 分 配 的 一 个 块 尾 释 放 一 种 资源 造 
成 的 (破坏 器 ”在 此 时 显得 特别 方便 )。 垃 圾 收集 器 是 在 C++ 基 础 上 的 
一 种 极 大 进步 ， 使 许多 编程 问题 消 弥 于 无 形 之 中 。 但 对 少数 几 个 垃圾 收 
集 胡 力 有 不 妹 的 问题 ， 它 却 是 不 大 适合 的 。 但 垃圾 收集 器 的 大 量 优点 也 
使 这 一 处 缺点 显得 微不足道 。 


(26) Java 内 建 了 对 多 线程 的 文 持 。 利 用 一 个 特殊 的 Thread 类 ， 我 们 可 通 
过 继承 创建 一 个 新 线程 〈 放 弃 了 run0) 方 法 ) 。 知 将 synchronized 〈 同 

步 ) 关键 字 作 为 方法 的 一 个 类 型 限制 符 人 使用， 相互 排斥 现象 会 在 对 象 这 
一 级 发 生 。 在 任何 给 定 的 时 间 ， 只 有 一 个 线程 能 使 用 一 个 对 象 的 
synchronized 方 法 。 在 男 一 方面 ， 一 个 synchronized 方 法 进入 以 后 ， 它 首 
先 会 “锁定 ”对 象 ， 防 止 其 他 任何 synchronized 方 法 再 使 用 那个 对 象 。 只 
有 退出 了 这 个 方法 ， 才 会 将 对 象 “解锁 ?。 在 线程 之 间 ， 我 们 仍然 要 负责 
SCALES ARAN IAP AL, FEC CASI aR? SS. BBVA 
synchronized 方 法 可 以 正常 运作 。 硅 线程 的 优先 等 级 相同 ， 则 时 间 的 “分 
厂 ” 不 能 得 到 保证 。 


(27) 我 们 不 是 象 Ct+ 那 样 控 制 声 明代 码 块 ， 而 是 将 访问 限定 符 
(public, privateflprotected) 置 入 每 个 类 成 员 的 定义 里 。 知 未 规定 一 
SERN” CHAD REW, MERUN KEH” (friendly) 。 这 意味 
独 同 一 个 包 里 的 其 他 元 素 也 可 以 访问 它 〈 相 当 于 它们 都 成 为 
C++ 的 “friends” 一 一 朋友 〉 ， 但 不 可 由 包 外 的 任何 元 素 访问 。 类 一 一 以 
及 类 内 的 每 个 方法 一 一 都 有 一 个 访问 限定 符 ， 决 定 它 是 耕 能 在 文件 的 外 
部 “可 见 ”?。private 关 键 字 通常 很 少 在 Java 中 使 用 ， 因 为 与 排斥 同一 个 包 
内 其 他 类 的 访问 相 比 , “友好 的 ?访问 通常 更 加 有 用 。 然 而 ， 在 多 线程 的 
环境 中 ， 对 private 的 恰当 运用 是 非常 重要 的 。Java 的 protected 关 键 字 意 
味 着 “可 由 继承 者 访问 ， 亦 可 由 包 内 其 他 元 素 访 问 ”*。 注 意 Java 没 有 与 



































C++ 的 protected 关 键 字 等 价 的 元 素 ， 后 者 意味 着 “只 能 由 继承 者 访 
问 ”( 以 前 可 用 “private protected” 实 现 这 个 日 的 ， 但 这 一 对 关键 字 的 组 合 
CRR T) o 


(28) 谋 套 的 类 。 在 Ct++ 中 ， 对 类 进行 构 侠 有 助 于 隐藏 名 称 ， 并 便于 代码 
的 组 织 《〈 但 C++ 的 “命名 空间 ?已 使 名 称 的 隐藏 显得 多 余 ) 。Java 的 “ 封 
装 ? 或 “打包 ?概念 等 价 于 C++ 的 命名 空间 ， 所 以 不 再 是 一 个 问题 。Java 
1.1 引 入 了 “内 部 类 ”的 概念 ， 筷 秘密 保持 指 同 外 部 类 的 一 个 句柄 一 一 创建 
内 部 类 对 象 的 时 候 需 要 用 到 。 这 意味 着 内 部 类 对 象 也 许 能 访问 外 部 类 对 
象 的 成 员 ， 毋 需 任 何 条 件 一 一 就 好 象 那些 成 员 直 接 隶 属于 内 部 类 对 象 一 
样 。 这 样 便 为 回调 问题 提供 了 一 个 更 优秀 的 方案 一 一 C++ 是 用 指 问 成 员 
的 指针 解决 的 。 


(a 由 于 存在 前 面 介绍 的 那 种 内 部 类 ， 所 以 Java 里 没有 指 癌 成 员 的 指 

















(30) Java FFTE“HRA” Cinline) FYE. Javan Vets VES AT RE RRA 
一 个 方法 ， 但 我 们 对 此 没有 更 多 的 控制 权力 。 在 Java 中 ， 可 为 一 个 方法 
使 用 final 关 键 字 ， 从 而 “建议 ”进行 通 入 操作 。 然 而 ， 髓 入 函数 对 于 
C++ 的 编译 器 来 说 也 只 是 一 种 建议 。 


(31) Java 中 的 继承 具有 与 C++ 相同 的 效果 ， 但 采用 的 语法 不 同 。Java 用 
extends 关 键 字 标 志 从 一 个 基础 类 的 继承 ， 并 用 super 关 键 字 指 出 准备 在 
基础 类 中 调用 的 方法 ， 它 与 我 们 当前 所 在 的 方法 具有 相同 的 名 字 OR 
而 ，Java 中 的 super 关 键 字 只 人 允许 我 们 访问 父 类 的 方法 一 一 亦 即 分 级 结构 
的 上 一 级 ) 。 通 过 在 C++ 中 设 定 基 础 类 的 作用 域 ， 我 们 可 访问 位 于 分 级 
结构 较 深 处 的 方法 。 亦 可 用 super 关 键 字 调用 基础 类 构建 器 。 正 如 早先 指 
出 的 那样 ， 所 有 类 最 终 都 会 从 Object 里 自动 继承 。 和 C++ 不 同 ， 不 存在 
明确 的 构建 器 初始 化 列表 。 但 编译 器 会 强迫 我 们 在 构建 器 主体 的 开头 进 
行 全 部 的 基础 类 初始 化 ， 而 且 不 允许 我 们 在 主体 的 后 面部 分 进行 这 一 工 
作 。 通 过 组 合 运 用 自动 初始 化 以 及 来 自 未 初始 化 对 象 句柄 的 异常 ， 成 员 
的 初始 化 可 得 到 有 效 的 保证 。 


public class Foo extends Bar { 











public Foo(String msg) { 


super(msg); // Calls base constructor 


} 
public baz(int i) { // Override 


super.baz(i); // Calls base method 


} 


(32) Java 中 的 继承 不 会 改变 基础 类 成 员 的 保护 级 别 。 我 们 不 能 在 Java 中 
指定 public，Pprivate 或 者 protected 继 承 ， 这 一 点 与 C++ 是 相同 的 。 此 外 ， 
在 衍生 类 中 的 优先 方法 不 能 减少 对 基础 类 方法 的 访问 。 例 如 ， 假 设 一 个 
成 员 在 基础 类 中 属于 public， 而 我 们 用 另 一 个 方法 代 蔡 了 它 ， 那 么 用 于 
替换 的 方法 也 必须 属于 public“〈 编 译 器 会 自动 检查 ) 。 


(33) Java 提 供 了 一 个 interface 关 键 字 ， 它 的 作用 是 创建 抽象 基础 类 的 一 个 
等 价 物 。 在 其 中 填充 抽象 方法 ， 且 没有 数据 成 员 。 这 样 一 来 ， 对 于 仅仅 
设计 成 一 个 接口 的 东西 ， 以 及 对 于 用 extends 关 键 字 在 现 有 功能 基础 上 的 
扩展 ， 两 者 之 间 便 产生 了 一 个 明显 的 差异 。 不 值得 用 abstract 关 键 字 产生 
一 种 类 似 的 效果 ， 因 为 我 们 不 能 创建 属于 那个 类 的 一 个 对 象 。 一 个 
abstract (FHA) 类 可 包 售 抽象 方法 〈 尽 管 并 不 要 求 在 它 里 面包 含 什么 东 
Po) ， 但 它 也 能 包含 用 于 具体 实现 的 代码 。 因 此 ， 它 被 限制 成 一 个 单一 
的 继承 。 通 过 与 接口 联合 使 用 ， 这 一 方案 避免 了 对 类 似 于 C++ 虚拟 基础 
类 那样 的 一 些 机 制 的 需要 。 


为 创建 可 进行 “ 例 示 ”( 即 创建 一 个 实例 ) 的 一 个 interface〈 接 口 ) 的 版 
本 ， 需 使 用 implements 关 键 字 。 它 的 语法 类 似 于 继承 的 语法 ， 如 下 所 
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public interface Face { 


public void smile(); 
} 
public class Baz extends Bar implements Face { 


public void smile( ) { 


System.out.printin("a warm smile"); 


(34) Java 中 没有 virtual 关 键 字 ， 因 为 所 有 非 static 方 法 都 肯定 会 用 到 动态 
绑 定 。 在 Java 中 ， 程 序 员 不 必 自 行 决定 是 否 使 用 动态 绑 定 。C++ 之 所 以 
采用 了 virtual， 是 由 于 我 们 对 性 能 进行 调整 的 时 候 ， 可 通过 将 其 省 略 ， 

从 而 获得 执行 效率 的 少量 提升 (或 者 换 句 话说 :“ 如 果 不 用 ， 就 没 必要 
为 它 付 出 代价 ”) 。virtual 经 党 会 造成 一 定 程度 的 混淆 ， 而 且 获 得 令 人 不 
快 的 结果 。final 关 键 字 为 性 能 的 调整 规定 了 一 些 范围 一 一 它 向 编译 器 指 
出 这 种 方法 不 能 被 取代 ， 所 以 它 的 范围 可 能 被 静态 约束 (而 且 成 为 租 入 
状态 ， 所 以 使 用 C++ 非 virtual 调 用 的 等 价 方式 ) 。 这 些 优化 工作 是 由 编 
译 器 完成 的 。 


(35) Java 不 提供 多 重 继承 机 制 CMD) ， 至 少 不 象 C++ 那样 做 。 与 
protected 类 似 ，MI 表 面 上 是 一 个 很 不 错 的 主意 ， 但 只 有 真正 面 对 一 个 特 
定 的 设计 问题 时 ， 才 知道 自己 需要 它 。 由 于 Java 使 用 的 是 “ 单 根 ” 分 级 结 
构 ， 所 以 只 有 在 极 少 的 场合 才 需 要 用 到 MI。interface 关 键 字 会 帮助 我 们 
自动 完成 多 个 接口 的 合并 工作 。 


(36) ”运行 期 的 类 型 标识 功能 与 C++ 极 为 相似 。 例 如 ， 为 获得 与 句柄 X 有 
关 的 信息 ， 可 使 用 下 述 代码 : 


X.getClass().getName(); 

为 进行 一 个 “类 型 安全 ”的 紧缩 造型 ， 可 使 用 : 

derived d = (derived)base; 

这 与 旧式 风格 的 C 造 型 是 一 样 的。 编译 费 会 自动 调用 动态 造型 机 制 ， 不 
要 求 使 用 额外 的 语法 。 尽 管 它 并 不 象 C++ 的 “new casts” ABFA A DE 
位 造型 的 优点 ， 但 Java 会 检查 使 用 情况 ， 并 丢弃 那些 “异常 "， 所 以 它 不 
会 象 C++ 那 样 允许 坏 造型 的 存在 。 


(37) Java 采 取 了 不 同 的 异常 控制 机 制 ， 因 为 此 时 已 经 不 存在 构建 器 。 可 
添加 一 个 finally 从 名 ， 强 制 执行 特定 的 语句 ， 以 便 进行 必要 的 清除 工 






































作 。Java 中 的 所 有 异常 都 是 从 基础 类 Throwable 里 继承 而 来 的 ， 所 以 可 确 
保 我 们 得 到 的 是 一 个 通用 接口 。 


public void f(Obj b) throws IOException { 


myresource mr = b.createResource(); 


try { 


mr .UseResource(); 


WY 


catch (MyException e) { 


// handle my exception 


WY 


catch (Throwable e) { 


// handle all other exceptions 


WY 


finally { 


mr.dispose(); // special cleanup 


(38) Java 的 异常 规范 比 C++ 的 出 色 得 多 。 丢 弃 一 个 错误 的 异常 后 ， 不 是 
象 C++ 那样 在 运行 期 间 调 用 一 个 函数 ，Java 肛 利 规 范 是 在 编译 期 间 检 查 
并 执行 的 。 除 此 以 外 ， 被 取代 的 方法 必须 遵守 那 一 方法 的 基础 类 版 本 的 
异常 规范 : 它们 可 丢 奔 指定 的 卉 间或 者 从 那些 异常 衍生 出 来 的 其 他 腊 

常 。 这 样 一 来 ， 我 们 最 终 得 到 的 是 更 为 “健壮 ”的 腊 常 控制 代码 。 


(39) ”Java 具 有 方法 过 载 的 能 力 ， 但 不 允许 运算 行 过 载 。String 类 不 能 
+ 和 += 运 算 符 连接 不 同 的 字 串 ， 而 且 String 表 达 式 使 用 自动 的 类 型 转换 ， 
但 那 是 一 种 特殊 的 内 建 情况 。 


(40) 通过 事先 的 约定 ，C++ 中 经 常 出 现 的 const 问 题 在 Java 里 已 得 到 了 控 
制 。 我 们 只 能 传递 指向 对 象 的 句柄 ， 本 地 副本 永远 不 会 为 我 们 上 自动 生 
成 。 若 希望 使 用 类 似 C++ 按 值 传递 那样 的 技术 ， 可 调用 clone()， 生 成 自 











变量 的 一 个 本 地 副本 (尽管 
革 ) 。 根 本 不 存在 被 日 动 调用 的 副本 构建 费 。 为 创建 一 个 编译 期 的 常数 
值 ， 可 和 象 下 面 这 样 编码 : 


static final int SIZE = 255 
static final int BSIZE = 8 * SIZE 


(41) 由 于 安全 方面 的 原因 ,“ 应 用 程序 "的 编程 也“ 程序 刻 " 的 编程 之 间 存 
在 着 显著 的 差异。 一 个 最 明显 的 问题 是 程序 片 不 允许 我 们 进行 磁盘 的 写 
操作 ， 因 为 这 样 做 会 造成 从 远程 站 点 下 载 的 、 不 明 来 历 的 程序 可 能 明 乱 
改写 我 们 的 磁盘 。 随 着 Java ”1.1 对 数字 签名 技术 的 引用 ， 这 一 情况 已 有 
所 改观 。 根 据 数字 签名 ， 我 们 可 确切 知道 一 个 程序 片 的 全 部 作者 ， 并 验 
证 他 们 是 否 已 获得 授权 。Java 1.2 会 进一步 增强 程序 片 的 能 力 。 


(42) 由 于 Java 在 某 些 场合 可 能 显得 限制 太 多 ， 所 以 有 时 不 愿 用 它 执 行 象 
直接 访问 硬件 这 样 的 重要 任务 。Java 解 决 这 个 问题 的 方案 是 “固有 方 

法 ”， 人 允许 我 们 调用 由 其 他 语言 写成 的 函数 (目前 只 支持 C 和 C++) 。 

样 一 来 ， 我 们 就 肯定 能 够 解决 与 平台 有 关 的 问题 (采用 一 种 不 可 移 和 的 
形式 ， 但 那些 代码 随后 会 被 隔离 起 来 ) 。 程 序 片 不 能 调用 固有 方法 ， 只 
有 应 用 程序 才 可 以 。 

(43) Java 提 供 对 注释 文档 的 内 建 支持 ， 所 以 源码 文件 也 可 以 包含 它们 自 
己 的 文档 。 通 过 一 个 单独 的 程序 ， 这 些 文档 信息 可 以 提取 出 来 ， 并 重新 
格式 化 成 HTML。 这 无 疑 是 文档 管理 及 应 用 的 极 大 进步 。 


(44) Java 包 含 了 一 些 标准 库 ， 用 于 完成 特定 的 任务 。C++ 则 依靠 一 些 非 
标准 的 、 由 其 他 厂商 提供 的 库 。 这 些 任务 包括 (或 不 久 就 要 包括 〉: 


mid: j 
国 数 据 库 连接 〈 通 过 JDBC ) 
四 多 线程 


四 分 布 式 对 象 〈( 通 过 RMI 和 CORBA) 


























四 压缩 


ae 而 且 非 常 标准 ， 所 以 能 极 大 加 快 应 用 程序 的 开发 
RJZ o 


(45) Java 1.1 包 含 了 Java Beans 标 准 ， 后 者 可 创建 在 可 视 编程 环境 中 使 用 
的 组 件 。 由 于 遵守 同样 的 标准 ， 所 以 可 视 组 件 能 够 在 所 有 厂商 的 开发 环 
境 中 使 用 。 由 于 我 们 并 不 依赖 一 家 厂商 的 方案 进行 可 视 组 件 的 设计 ， 所 
以 组 件 的 选择 余地 会 加 大 ， 并 可 提高 组 件 的 效能 。 除 此 之 外 ，Java 
Beans 的 设计 非常 简单 ， 便 于 程序 员 理解 ， 而 那些 由 不 同 的 广 商 开发 的 
专用 组 件 框架 则 要 求 进行 更 深入 的 学 习 。 

(46) ÆW Javah] WAM, MeL -KEE KES MAIN 
要 正好 在 使 用 一 个 句柄 之 前 进行 。 根 据 Java 的 设计 规范 ， 只 是 说 寞 第 必 
须 以 某 种 形式 丢弃 。 许 多 C++ 运 行 期 系统 也 能 丢弃 那些 由 于 指针 错误 造 
成 的 异常 。 

(47) Java 通 常 显得 更 为 健壮 ， 为 此 采取 的 手段 如 下 : 

四 对 象 句 顶 初始 化 成 null (一 个 关键 字 ) 

别人 急 顶 肯定 会 得 到 检查 ， 并 在 出 错时 丢弃 异常 

加 所 有 数组 访问 都 会 得 到 检查 ， 及 时 发 现 边界 违例 情况 

加 自动 垃圾 收集 ， 防 止 出 现 内 存 漏洞 

加 明确 、“ 傻 瓜 式 ”的 异常 控制 机 制 

四 为 多 线程 提供 了 简单 的 语言 支持 

加 对 网 络 程序 片 进 行 字 节 人 码 校 验 














附录 C Java 编 程 规则 


本 附录 包含 了 大 量 有 用 的 建议 ， 帮 助 大 家 进行 低级 程序 设计 ， 并 提供 了 
代码 编写 的 一 般 性 指导 : 

O 类 名 前 字 母 应 该 大 写 。 字 段 、 方 法 以 及 对 象 ( 人 句柄) 的 首 字 母 应 小 
写 。 对 于 所 有 标识 符 ， 其 中 包含 的 所 有 单词 都 应 紧 徘 在 一 起 ， 而 且 大 写 
中 间 单 词 的 首 字 母 。 例 如 ; 


ThisIsAClassName 








thislsMethodOrFieldName 


在 在 定义 中 出 现 了 第 数 初 始 化 字符 ， 则 大 写 static final 基 本 类 型 标识 符 中 
的 所 有 字母 。 这 样 便 可 标志 出 它们 属于 编译 期 的 常数 。 


Java, (Package) 属于 一 种 特殊 情况 : 它们 全 都 是 小 写字 母 ， 即 便 中 间 
的 单词 亦 是 如 此 。 对 于 域名 扩展 名 称 ， 如 com，org，net 或 者 edu 等 ， 全 
部 都 应 小 写 (这 也 是 Java 1.1 和 Java 1.2 的 区 别 之 一 ) 。 


(2) 为 了 常规 用 途 而 创建 一 个 类 时 ， 请 采取 “经 典 形式 ”， 并 包含 对 下 述 
元 素 的 定义 : 











equals() 

hashCode() 

toString() 

clone() (implement Cloneable ) 


implement Serializable 


(3) 对 于 目 己 创建 的 每 一 个 类 ， 都 考虑 置 入 一 个 main0， 其 中 包含 了 用 于 
测试 那个 类 的 代码 。 为 使 用 一 个 项 目 中 的 类 ， 我 们 没 必 要 删除 测试 代 
码 。 硅 进行 了 任何 形式 的 改动 ， 可 方便 地 返回 测试 。 这 些 代码 也 可 作为 
如 何 使 用 类 的 一 个 示例 使 用 。 


(4) ”应 将 方法 设计 成 简要 的 、 功 能 性 单元 ， 用 它 描述 和 实现 一 个 不 连续 
的 类 接口 部 分 。 理 想 情 况 下 ， 方 法 应 简明 扼要 。 知 长 度 很 大 ， 可 考虑 通 
过 茶 种 方式 将 其 分 割 成 较 短 的 几 个 方法 。 这 样 做 也 便于 类 内 代码 的 重复 
使 用 《有些 时 候 ， 方 法 必须 非常 大 ， 但 它们 仍 应 只 做 同样 的 一 件 事 


情 ) 。 


(5) 设计 一 个 类 时 ， 请 设身处地 为 客户 程序 员 考 虑 一 下 《类 的 使 用 方法 
应 该 是 非常 明确 的 ) 。 然 后 ， 再 设 吴 处 地 为 管理 代码 的 人 考虑 一 下 《了 预 
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Pre 


(6) 使 类 尽 可 能 短小 精 悍 ， 而 且 只 解决 一 个 特定 的 问题 。 下 面 是 对 关 设 


四 一 个 复杂 的 开关 语句 : 考虑 采用 “多 形 ” 机 制 


四 数量 众多 的 方法 涉及 到 类 型 送别 极 大 的 操作 : 考虑 用 几 个 类 来 分 别 实 
现 


加 许多 成 员 变 量 在 特征 上 有 很 大 的 差别 : 考虑 使 用 几 个 类 


(7) 让 一 切 东 西 都 尽 可 能 地 “私有 ” private。 可 使 库 的 某 一 部 分 “公共 
化 ”( 一 个 方法 、 类 或 者 一 个 字段 等 等 ) ， 就 永远 不 能 把 它 拿 出 。 若 强 
行 拿 出， 就 可 能 破坏 其 他 人 现 有 的 代码 ， 使 他 们 不 得 不 重新 编号 和 设 

计 。 知 只 公布 目 己 必须 公布 的 ， 束 可 放心 大 胆 地 改变 其 他 任何 东西 。 在 
多 线程 环境 中 ， 隐 私 是 特别 重要 的 一 个 因素 只 有 private 字 段 才能 在 
非 同步 使 用 的 情况 下 受到 保护 。 


(8)” 诬 惕 “巨大 对 象 综合 症 *"。 对 一 些 习 惯 于 顺序 编程 思维 、 且 初 涉 OOP 
领域 的 新 手 ， 往 往 训 欢 先 写 一 个 顺序 执行 的 程序 ， 再 把 它 租 入 一 个 或 两 
个 巨大 的 对 象 里。 根据 编程 原理 ， 对 象 表达 的 应 该 是 应 用 程序 的 概念 ， 
而 非 应 用 程序 本 里。 


(9) 奉 不 得 已 进行 一 些 不 太 雅 观 的 编程 ， 至 少 应 该 把 那些 代码 置 于 一 个 
类 的 内 部 。 


(10) 任何 时 候 只 要 发 现 类 与 类 之 间 结 合 得 非常 紧密 ， 就 需要 考虑 是 否 采 
用 内 部 类 ， 从 而 改善 编码 及 维护 工作 (参见 第 14 间 14.1.2 小 节 的 “用 内 部 























类 改进 代码 ”) 。 


(11) 尽 可 能 细致 地 加 上 注释 ， 并 用 javadoc 注 释文 档 语法 生成 目 己 的 程序 
文档 。 


(12) 避免 使 用 “魔术 数字 ”， 这 些 数字 很 难 与 代码 很 好 地 配合 。 如 以 后 需 
要 修改 它 ， 无 疑 会 成 为 一 场 墨 梦 ， 因 为 根本 不 知道 <100? 到 底 是 指 “ 数 组 
大 小 ”还 是 “其 他 全 然 不 同 的 东西 "?。 所 以 ,我们 应 创建 一 个 第 数 ， 并 为 

其 使 用 具有 说 服 力 的 描述 性 名 称 ， 并 在 整个 程序 中 都 采用 常数 标识 符 。 

这 样 可 使 程序 更 易 理解 以 及 更 易 维护 。 


(13) 涉及 构建 器 和 有 弄 第 的 时 候 ， 通 单 希 望 重新 丢弃 在 构建 器 中 捕获 的 任 
何 异 常 一 一 如 果 它 造成 了 那个 对 象 的 创建 失败 。 这 样 一 来 ， 调 用 者 就 不 
会 以 为 那个 对 象 已 正确 地 创建 ， 从 而 盲目 地 继续 。 


(14) 当 客 户 程 序 员 用 完 对 象 以 后 ， 奋 你 的 类 要 求 进行 任何 清除 工作 ， 可 
考虑 将 清除 代码 置 于 一 个 展 好 定义 的 方法 里 ， 采 用 类 似 于 cleanupO 这 样 
的 名 字 ， 明 确 表明 自己 的 用 途 。 除 此 以 外 ， 可 在 类 内 放置 一 个 

boolean (H/K) 标记 ， 指 出 对 象 是 否 已 被 清除 。 在 类 的 finalize() 方 法 
里 ， 请 确定 对 象 已 被 清除 ， 并 已 丢弃 了 从 RuntimeException 继 承 的 一 个 
类 如果 还 没有 的 话 ) ， 从 而 指出 一 个 编程 错误 。 在 采取 象 这 样 的 方案 
之 前 ， 请 确定 finalize() 能 够 在 自己 的 系统 中 工作 (可 能 需要 调用 
System.runFinalizersOnExit(true)， 从 而 确保 这 一 行为 ) 。 


(15) 在 一 个 特定 的 作用 域内 ， 若 一 个 对 象 必 须 清除 〈 非 由 垃圾 收集 机 制 
处 理 ) ， 请 采用 下 述 方法 : 初始 化 对 象 ， 辱 成 功 ， 则 立即 进入 一 个 含 
finally 从 句 的 try 块 ， 开 始 清除 工作 。 


(16) 知 在 初始 化 过 程 中 需要 履 盖 〈 取 消 ) finaljize0， 请 记 住 调用 
super.finalize() 〈 知 Object 属于 我 们 的 直接 超 类 ， 则 无 此 必要 ) 。 在 对 
finalizeO 进 行 覆 盖 的 过 程 中 ， 对 super.finalize0 的 调用 应 属于 最 后 一 个 行 
Se eae ， 这 样 可 确保 在 需要 基础 类 组 件 的 时 候 它 们 依 
RA RM o 


(17) 创建 大 小 固定 的 对 象 集合 时 ， 请 将 它们 传输 到 一 个 数组 〈 知 准备 从 
一 个 方法 里 返回 这 个 集合 ， 更 应 如 此 操作 ) 。 这 样 一 来 ， 我 们 就 可 享受 
到 数组 在 编译 期 进行 类 型 检查 的 好 处 。 此 外 ， 为 使 用 它们 ， 数 组 的 接收 
者 也 许 并 不 需要 将 对 象 < 造型 ?到 数组 里 。 

















(18) 尽量 使 用 interfaces， 不 要 使 用 abstract 类 。 知 已 知 某 样 东 西 准备 成 为 
一 个 基础 类 ， 那 么 第 一 个 选择 应 是 将 其 变 成 一 个 interface (ZO) 。 只 

有 在 不 得 不 使 用 方法 定义 或 者 成 员 变 量 的 时 候 ， 才 需要 将 其 变 成 一 个 

abstract (FHA) 类 。 接 口 主 要 描述 了 客户 和 希望 做 什么 事情 ， 而 一 个 类 则 
致力 于 《或 允许 ) 具体 的 实施 细节 。 

(19) 在 构建 器 内 部 ， 只 进行 那些 将 对 象 设 为 正确 状态 所 需 的 工作 。 尽 可 
能 地 避免 调用 其 他 方法 ， 因 为 那些 方法 可 能 被 其 他 人 和 窗 亲 或 取消 ， 从 而 
在 构建 过 程 中 产生 不 可 预知 的 结果 (参见 第 7 章 的 详细 说 明 )。 


(20) 对 象 不 应 只 是 简单 地 容纳 一 些 数 据 ; 它们 的 行为 也 应 得 到 民 好 的 定 
义 。 











(21) 在 现成 类 的 基础 上 创建 新 类 时 ， 请 首先 选择 “新 建 ?或 “创作 >”。 只 有 
目 己 的 设计 要 求 必 须 继 承 时 ， 才 应 考虑 这 方面 的 问题 。 奋 在 本 来 允许 新 
建 的 场合 使 用 了 继承 ， 则 整个 设计 会 变 得 没有 必要 地 复杂 。 


(22) 用 继承 及 方法 罗 盖 来 表示 行为 间 的 兰 异 ， 而 用 字段 表示 状态 间 的 区 
别 。 一 个 非常 极端 的 例子 是 通过 对 不 同类 的 继承 来 表示 颜色 ， 这 是 绝对 
应 该 避免 的 : 应 直接 使 用 一 个 “颜色 ”字段 。 


(23) 为 避免 编程 时 遇 到 麻烦 ， 请 保证 在 自己 类 路 径 指 到 的 任何 地 方 ， 每 
个 名 字 都 仅 对 应 一 个 类 。 否 则 ， 编 译 占 可 能 先 找到 同名 的 妨 一 个 类 ， 并 
报告 出 错 消 妃 。 知 怀疑 目 己 碰 到 了 类 路 径 问 题 ， 请 试 试 在 类 路 径 的 每 一 
个 起 点 ， 搜 索 一 下 同名 的 .class 文 件 。 


(24) 在 Java 1.1 AWT 中 使 用 事件 “适配器 ”时 ， 特 别 容 易 伴 到 一 个 陷阱 。 
知 履 盖 了 某 个 适配器 方法 ， 同 时 拼写 方法 没有 特别 讲究 ， 最 后 的 结果 就 
是 新 添加 一 个 方法 ， 而 不 是 履 盖 现成 方法 。 然 而 ， 由 于 这 样 做 是 完全 合 
法 的 ， 所 以 不 会 从 编译 器 或 运行 期 系统 获得 任何 出 错 提示 一 一 只 不 过 代 
码 的 工作 就 变 得 不 正常 了 。 


(25) 用 合理 的 设计 方案 消除 “ 仿 功 能 ”。 也 就 是 说 ， 假 各 只 需要 创建 类 的 
一 个 对 象 ， 束 不 要 提前 限制 自己 使 用 应 用 程序 ， 并 加 上 一 条 “只 生成 其 
中 一 个 ”注释 。 请 考虑 将 其 封装 成 一 个 “独生子 ”的 形式 。 有 在 主 程序 里 
有 大 量 散 乱 的 代码 ， 用 于 创建 自己 的 对 象 ， 请 考虑 采纳 一 种 创造 性 的 方 
案 ， 将 些 代 码 封装 起 来 。 






































(26) ”和 警惕 “分 析 次 痪 >"。 请 记 住 ， 无 论 如 何 都 要 提前 了 解 整个 项 目的 状 
况 ， 再 去 考察 其 中 的 细节 。 由 于 把 握 了 全 局 ， 可 快速 认识 上 自己 未 知 的 一 
些 因素 ， 防 止 在 考察 细节 的 时 候 陷 入 “ 死 逻 辑 ” 中 。 


(27) 警惕 “过 早 优 化 ”。 首 先 让 它 运行 起 来 ， 再 考虑 变 得 更 快 一 一 但 只 有 
在 上 自己 必须 这 样 做 、 而 且 经 证 实在 某 部 分 代码 中 的 确 存在 一 个 性 能 瓶颈 
的 时 候 ， 才 应 进行 优化 。 除 非 用 专门 的 工具 分 析 瓶 宽 ， 否 则 很 有 可 能 是 
在 浪费 自己 的 时 间 。 性 能 提升 的 隐 舍 代价 是 自己 的 代码 变 得 难于 理解 ， 
而 且 难 于 维护 。 


(28) 请 记 住 ， 阅 读 代 码 的 时 间 比 写 代 码 的 时 间 多 得 多 。 思 路 清晰 的 设计 
可 获得 易于 理解 的 程序 ， 但 注释 、 细 致 的 解释 以 及 一 些 示 例 往往 具有 不 
可 估量 的 价值 。 无 论 对 你 目 己 ， 还 是 对 后 来 的 人 ， 它 们 都 是 相当 重要 

的 。 如 对 此 仍 有 怀疑 ， 那 么 请 试想 自己 试图 从 联机 Java 文 档 里 找 出 有 用 
言 轧 时 碰 到 的 挫折 ， 这 样 或 许 能 将 你 说 服 。 


(29) 如 认为 自己 已 进行 了 恨 好 的 分 析 、 设 计 或 者 实施 ， 那 么 请 稍微 更 换 
一 下 思维 角度 。 试 试 邀请 一 些 外 来 人 士 一 一 并 不 一 定 是 专家 ,但 可 以 是 
来 日 本 公司 其 他 部 门 的 人 。 请 他 们 用 完全 新 鲜 的 眼光 考察 你 的 工作 ， 看 
看 是 否 能 找 出 你 一 度 熟 视 无 睹 的 问题 。 采 取 这 种 方式 ， 往 往 能 在 最 适合 
修改 的 阶段 找 出 一 些 关 键 性 的 问题 ， 避 免 产 品 发 行 后 再 解决 问题 而 造成 
的 金钱 及 精力 方面 的 损失 。 


(30) 展 好 的 设计 能 带 来 最 大 的 回报 。 简 言 之 ， 对 于 一 个 特定 的 问题 ， 通 
常会 花 较 长 的 时 间 才 能 找到 一 种 最 恰当 的 解决 方案 。 但 一 旦 找到 了 正确 
的 方法 ， 以 后 的 工作 束 轻 松 多 了 ， 再 也 不 用 经 历数 小 时 、 数 天 或 者 数 月 
的 痛 否 挣扎 。 我 们 的 努力 工作 会 带 来 最 大 的 回报 《〈 甚 全 无 可 估量 ) o M 
且 由 于 自己 倾注 了 大 量 心血 ， 最 终 获 得 一 个 出 色 的 设计 方案 ， 成 功 的 快 
0 0 那样 做 往往 得 不 偿 


(31) 可 在 Web 上 找到 大 量 的 编程 参考 资源 ， 甚 至 包括 大 量 新 闻 组 、 讨 论 
组 、 邮 和 寄 列 表 等 。 下 面 这 个 地 方 提供 了 大 量 有 益 的 链接 : 
































http://www.ulb.ac.be/esp/ip-Links/Java/joodcs/mm-WebBiblio.html 


附录 DD 性 能 


“本 附录 由 Joe Sharp 投 稿 ， 并 获得 他 的 同意 在 这 儿 转 载 。 请 联系 
SharpJoe@aol.com” 


Java 语 言 特别 强调 准确 性 ， 但 可 靠 的 行为 要 以 性 能 作为 代价 。 这 一 特点 
反映 在 自动 收集 垃圾 、 严 格 的 运行 期 检查 、 完 整 的 字 节 码 检 查 以 及 保守 
的 运行 期 同步 等 等 方面 。 对 一 个 解释 型 的 虚拟 机 来 说 ， 由 于 目前 有 大 量 
平台 可 供 挑选 ， 所 以 进一步 阻 但 了 性 能 的 发 挥 。 


“ 先 做 完 它 ， 再 逐步 完善 。 邓 好 需要 改进 的 地 方 通常 不 会 太 多 。”(Steve 
McConnell 的 《About performance) [16]) 


本 附录 的 守则 束 是 指导 大 家 寻找 和 优化 “需要 完善 的 那 一 部 分 ”。 
D.1 基本 方法 
只 有 正确 和 完整 地 检测 了 程序 后 ， 再 可 着 手 解 决 性 能 方面 的 问题 : 


(1) 在 现实 环境 中 检测 程序 的 性 能 。 帮 符合 要 求 ， 则 目标 达到 。 硅 不 符 
合 ， 则 转 到 下 一 步 。 


(2) 寻找 最 致命 的 性 能 敌人 席 。 这 也 许 要 求 一 定 的 技巧 ， 但 所 有 努力 都 不 
会 白费 。 如 简单 地 猜测 瓶 瑞 所 在 ， 并 试图 进行 优化 ， 那 么 可 能 是 白花 时 
间 。 


(3) 运用 本 附录 介绍 的 提速 技术 ， 然 后 返回 步骤 1。 


AVES AAR EA, FRAME NEB HEY M. Donald Knuth[9] 
改进 过 一 个 程序 ， 那 个 程序 把 50% 的 时 间 都 花 在 约 4% 的 代码 量 上 。 在 
仅 一 个 工作 小 时 里 ， 他 修改 了 几 行 代码 ， 使 程序 的 执行 速度 倍增 。 此 
时 ， 寿 将 时 间 继 续 投入 到 剩余 代码 的 修改 上 ， 那 么 只 会 得 不 傍 失 。 
Knuth 在 编程 界 有 一 句 名 言 :“ 过 早 的 优化 是 一 切 麻烦 的 根 

W” (Premature optimization is the root of all evil) 。 最 明 乔 的 做 法 是 抑制 
过 早 优 化 的 冲动 ， 因 为 那样 做 可 能 遗漏 多 种 有 用 的 编程 技术 ， 造 成 代码 
更 难 理解 和 操控 ， 并 需 更 大 的 精力 进行 维护 。 




















D.2 FIRM 

为 找 出 最 影 啊 程序 性 能 的 瓶颈 ， 可 采取 下 述 几 种 方法 : 
D.2.1 安插 自己 的 测试 代码 

插入 下 述 “ 显 式 ” 计 时 代码 ， 对 程序 进行 评测 : 





long start = System.currentTimeMillis(); 
// 要 计时 的 运算 代码 放 在 这 儿 
long time = System.currentTimeMillis() - start; 


利用 System.out.println()， 让 一 种 不 常用 到 的 方法 将 累积 时 间 打 印 到 控制 
台 窗 口 。 由 于 一 旦 出 错 ， 编 译 器 会 将 其 忽略 ， 所 以 可 用 一 个 “静态 最 终 
布尔 值 ”(Static final boolean) 打开 或 关闭 计时 ， 使 代码 能 放心 留 在 最 
终 发 行 的 程序 里 ， 这 样 任 何 时 候 都 可 以 使 来 应 急 。 尽 管 还 可 以 选用 更 复 
杂 的 评测 手段 ， 但 知 仅仅 为 了 量度 一 个 特定 任务 的 执行 时 间 ， 这 无 疑 是 
最 简便 的 方法 。 


System.currentTimeMillisO 返 回 的 时 间 以 千 分 之 一 秒 〈1 毫 秒 ) 为 单位 。 
然而 ， 有 些 系统 的 时 间 精 度 低 于 1 坚 秒 〈 如 Windows PC) ， 上 所 以 需要 重 
复 n 次 ， 再 将 总 时 间 除 以 n， 获 得 准确 的 时 间 。 


D.2.2 JDK 性 能 评测 [2] 

JDK 配 套 提 供 了 一 个 内 建 的 评测 程序 ， 能 跟踪 花 在 每 个 例 程 上 的 时 间 ， 
并 将 评测 结果 写 入 一 个 文件 。 不 幸 的 是 ，JDK 评 测 器 并 不 稳定 。 它 在 
JDK 1.1.1 中 能 正常 工作 ， 但 在 后 续 版 本 中 却 非常 不 稳定 。 


ee ， 请 在 调用 Java 解 释 右 的 未 优化 版 本 时 加 上 -prof 选 项 。 
MUN : 





java_g -prof myClass 
或 加 上 一 个 程序 片 (Applet) : 


java_g -prof sun.applet.AppletViewer applet.html 


理解 评测 程序 的 输出 信息 并 不 容易 。 事 实 上 ， 在 JDK 1.0 中 ， 它 居然 将 
方法 名 称 截 短 为 30 字 符 。 所 以 可 能 无 法 区 分 出 某 些 方法 。 然 而 ， 帮 您 用 
的 平台 确实 能 支持 -prof 选 项 ， 那 么 可 试 试 Vladimir Bulatov 
的 “HyperPorf”[3] 或 者 Greg White 的 “ProfileViewer" 来 解释 一 下 结果 。 


D.2.3 特殊 工具 

如 有 果 想 随时 跟 上 性 能 优化 工具 的 潮流 ， 最 好 的 方法 就 是 作 一 些 Web 站 点 
的 常客 。 比 如 由 Jonathan ”Hardwick 制 作 的 “Tools for Optimizing 
Java”(〈Java 优 化 工具 ) 网 站 : 

http:/www.cs.cmu.edu/~jch/java/tools.html 

D.2.4 性 能 评测 的 技巧 


四 由 于 评测 时 要 用 到 系统 时 钟 ， 所 以 当时 不 要 运行 其 他 任何 进程 或 应 用 
程序 ， 以 免 影响 测试 结 


加 如 对 自己 的 程序 进行 了 修改 ， 并 试图 (至 少 在 开发 平台 上 ) 改善 它 的 
性 能 ， 那 么 在 修改 前 后 应 分 别 测试 一 下 代码 的 执行 时 间 。 


四 尽量 在 完全 一 致 的 环境 中 进行 每 一 次 时 间 测 试 。 


四 如 果 可 能 ， 应 设计 一 个 不 依赖 任何 用 户 输入 的 测试 ， 避 免 用 户 的 不 同 
反应 导致 结果 出 现 误差 。 


D.3 提速 方法 


现在 ， 关 键 的 性 能 瓶颈 应 已 隔离 出 来 。 接 下 来 ， 可 对 其 应 用 两 种 类型 的 
优化 : 常规 手段 以 及 依赖 Java 语 言 。 


D.3.1 常规 手段 


通常 ， 一 个 有 效 的 提速 方法 是 用 更 现实 的 方式 重新 定义 程序 。 例 如 ， 在 
(Programming Pearls》【〔 编 程 拾 贝 ) 一 书 中 [14]，Bentley 利 用 了 一 段 小 
说 数据 描写 ， 它 可 以 生成 速度 非常 快 、 而 且 非 常 精简 的 拼写 检查 器 ， 从 
而 介绍 了 Doug ”McIlroy 对 有 疯 语 语言 的 表述 。 除 此 以 外 ， 与 其 他 方法 相 
比 ， 更 好 的 算法 也 许 能 带 来 更 大 的 性 能 提升 特别 是 在 数据 集 的 尺寸 
越 来 越 大 的 时 候 。 欲 了 解 这 些 常 规 手 段 的 详情 ， 请 参考 本 附录 末尾 

















的 “一 般 书 籍 ? 请 单 。 

D.3.2 依赖 语言 的 方法 

为 进行 客观 的 分 析 ， 最 好 明确 掌握 各 种 运算 的 执行 时 间 。 这 样 一 来 ， 得 
到 的 结果 可 独立 于 当前 使 用 的 计算 机 一 一 通过 除 以 花 在 本 地 赋值 上 的 时 
间 ， 最 后 得 到 的 就 是 “标准 时 间 ”。 

运算 示例 标准 时 间 

本 地 赋值 i=n; 1.0 


实例 赋值 this.i=n; 1.2 





int 增 值 i++; 1.5 

byte 增 值 b++; 2.0 

short 增 值 s++; 2.0 

float 增 值 f++; 2.0 

double 增 值 d++; 2.0 

空 循环 while(true) n++; 2.0 
三 元 表达 式 (x<0) ?-x : x 2.2 
算术 调用 Math.abs(x); 2.5 
数组 赋值 a[0] = n; 2.7 

long 增 值 1++; 3.5 

方法 调用 funct(); 5.9 
throw 或 catch 异 常 try{ throw e; } 或 catch(e){} 320 


同步 方法 调用 synchMehod(); 570 


新 建 对 象 new Object(); 980 

新 建 数 组 new int[10]; 3100 

通过 自己 的 系统 (如 我 的 Pentium 200 Pro, Netscape 3 及 JDK 1.1.5) ， 
这 些 相 对 时 间 同 大 家 揭示 出 : 新 建 对 象 和 数组 会 造成 最 沉重 的 开销 ， 同 
步 会 造成 比较 沉重 的 开销 ， 而 一 次 不 同步 的 方法 调用 会 造成 适度 的 开 
销 。 参 考 资 源 [5] 和 [6] 为 大 家 总 结 了 测量 用 程序 片 的 Web 地 址 ， 可 到 自 
己 的 机 器 上 运行 它们 。 

1. 常规 修改 


下 面 是 加 快 Java 程 订 关 键 部 分 执行 速度 的 一 些 常 规 操作 建议 注意 对 比 
修改 前 后 的 测试 结果 )。 


将 … 修改 成 … 理由 

接口 抽象 类 《只 需 一 个 父 时 ) 接口 的 多 个 继承 会 妨碍 性 能 的 优化 

非 本 地 或 数组 循环 变量 本 地 循环 变量 根据 前 表 的 耗 时 比较 ， 一 次 实例 
整数 赋值 的 时 间 是 本 地 整数 赋值 时 间 的 1.2 倍 ， 但 数组 赋值 的 时 间 是 本 
地 整数 赋值 的 2.7 倍 

链接 列表 (固定 尺寸 ) ”保存 丢弃 的 链接 项 目 ， 或 将 列表 蔡 换 成 一 个 循 
环 数组 (大 致知 道 尺寸 ) 每 新 建 一 个 对 象 ， 都 相当 于 本 地 赋值 980 次 。 
参考 “重复 利用 对 象 ”( 下 一 节 ) 、Van Wyk[12] p.87 以 及 Bentley[15] p.81 
X/2〈 或 2 的 任意 次 虹 ) X>>2《〈 或 2 的 任意 次 需 ) 使 用 更 快 的 便 件 指令 
D.3.3 特殊 情况 

BSN: 字 串 连接 运算 符 + 看 似 简单 ， 但 实际 需要 消耗 大 量 系 统 资 


源 。 编 译 器 可 高 效 地 连接 字 串 ， 但 变量 字 串 却 要 求 可 观 的 处 理 器 时 间 。 
例如 ， 假 设 s 和 t 是 字 串 变量 : 





System.out.printIn("heading" + s + "trailer" + t); 


上 述 语句 要 求 新 建 一 个 StringBuffer〈 字 串 缓冲 ) ， 追 加 自 变 量 ， 然 后 用 
toString0) 将 结果 转换 回 一 个 字 串 。 因 此 ， 无 论 磁盘 空间 还 是 处 理 絮 时 


间 ， 都 会 受到 严重 消耗 。 知 准备 退 加 多 个 字 溃 ， 则 可 考虑 直接 使 用 一 个 
字 串 绥 冲 一 一 特别 是 能 在 一 个 循环 里 重复 利用 它 的 时 候 。 通 过 在 每 次 循 
环 里 禁止 新 建 一 个 字 串 缓冲 ， 可 市 省 980 单 位 的 对 象 创建 时 间 (如 前 所 
R) 。 利 用 substringO0 以 及 其 他 字 串 方法 ， 可 进一步 地 改善 性 能 。 如 采 
可 行 ， 字 符 数 组 的 速度 甚至 能 够 更 快 。 也 要 注意 由 于 同步 的 关系 ， 所 以 
StringTokenizer 会 造成 较 大 的 开销 。 


加 同步 :在 JDK 解 释 嚣 中， 调用 同步 方法 通常 会 比 调用 不 同步 方法 慢 10 
倍 。 经 JIT 编 译 器 处 理 后 ， 这 一 性 能 上 的 差距 提升 到 50 到 100 倍 (注意 前 
表 总 结 的 时 间 显 示 出 要 慢 97 倍 ) 。 所 以 要 尽 可 能 避免 使 用 同步 方法 
若 不 能 避免 ， 方 法 的 同步 也 要 比 代 码 块 的 同步 稍 快 一 些 。 


四 重复 利用 对 象 : 要 花 很 长 的 时 间 来 新 建 一 个 对 象 〈 根 据 前 表 总 结 的 时 
闻 ， 对 象 的 新 建 时 间 是 赋值 时 间 的 980 倍 ， 而 新 建 一 个 小 数组 的 时 间 是 
赋值 时 间 的 3100 倍 ) 。 因 此 ， 最 明智 的 做 法 是 保存 和 更 新 老 对 象 的 字 
段 ， 而 不 是 创建 一 个 新 对 象 。 例 如 ， 不 要 在 自己 的 paint(0) 方 法 中 新 建 一 
个 Font 对 象 。 相 反 ， 应 将 其 声明 成 实例 对 象 ， 再 初始 化 一 次 。 在 这 以 
后 ， 可 在 paint(0 里 需要 的 时 候 随 时 进行 更 新 。 参 见 Bentley 编 著 的 《编程 
拾 贝 》，Pp.81[15]。 


esi: 只 有 在 不 正 币 的 情况 下 ， 才 应 放 奔 异 币 处 理 模 块 。 什 么 才 叫 “不 
正常 * 呢 ?这 通常 是 指 程序 过 到 了 问题 ， 而 这 一 般 是 不 愿 见 到 的 ， 所 以 
性 能 不 再 成 为 优先 考虑 的 目标 。 进 行 优化 时 ， 将 小 的 “try-catch” 块 合并 
到 一 起 。 由 于 这 些 块 将 代码 分 割 成 小 的 、 各 自 独 立 的 片断 ， 所 以 会 妨碍 
编译 器 进行 优化 。 力 一 方面 ， 耕 过 份 热 束 于 删除 异常 处 理 和 模块， 也 可 能 
造成 代码 健壮 程度 的 下 降 。 


mAb: 首先 ，Java 1.0 和 1.1 的 标准 “ 散 列 表 ”(Hashtable) 类 需要 造 
型 以 及 特别 消耗 系统 资源 的 同步 处 理 〈570 单 位 的 赋值 时 间 ) 。 其 次 ， 
早期 的 JDK 库 不 能 自动 决定 最 佳 的 表格 尺寸 。 最 后 ， 散 列 函数 应 针对 实 
际 使 用 项 (Key) 的 特征 设计 。 考 虑 到 所 有 这 些 原 因 ， 我 们 可 特别 设计 
一 个 散 列 类 ， 令 其 与 特定 的 应 用 程序 配合 ， 从 而 改善 常规 散 列 表 的 性 
能 。 注 意 Java 1.2 集 合 库 的 散 列 映射 (HashMap) 具有 更 大 的 灵活 性 ， 而 
且 不 会 自动 同步 。 


EDEN: 只 有 在 方法 属于 final CR) . private CHH) 或 
static (BRAS) INTL, Javani BeN x SIE. MASEL 
下 ， 还 要 求 它 绝对 不 可 以 有 局 部 变量 。 知 代码 花 大 量 时 间 调 用 一 个 不 含 



































上 述 任何 属性 的 方法 ， 那 么 请 考虑 为 其 编写 一 个 “final* 版 本 。 


mO: 应 尺 可 能 使 用 缓冲 。 否 则 ， 最 终 也 许 就 是 一 次 仅 输 入 / 输出 一 个 
字 节 的 有 恶果。 注意 JDK 1.0 的 IO 类 采用 了 大 量 同 步 措施 ， 所 以 右 使 用 象 
readFullyO 这 样 的 一 个 “大 批量 调用， 然后 由 目 己 解释 数据 ， 就 可 获得 
更 佳 的 性 能 。 也 要 注意 Java 1.1 的 “reader*" 和 “writer”* 类 已 针对 性 能 进行 了 
优化 。 


加 造型 和 实例 : 造型 会 耗 去 2 到 200 个 单位 的 赋值 时 间 。 开 销 更 大 的 甚至 
o (遗传 ) 结构 。 其 他 高 代价 的 操作 会 损失 和 恢复 更 低层 结 
义 的 能 


eR: 利用 剪 切 技术 ， 减 少 在 repaintO 中 的 工作 量 ; 倍增 缓冲 区 ， 提 高 
接收 速度 ; 同时 利用 图 形 压缩 技术 ， 纵 短 下载 时 间 。 来 自 JavaWorld 
的 “Java Applets” 以 及 来 自 Sun 的 “Performing Animation” 是 两 个 很 好 的 教 
程 。 请 记 着 使 用 最 贴切 的 命令 。 例 如 ， 为 根据 一 系列 点 男 一 个 多 边 形 ， 
和 drawLine() 相 比 ，drawPolygon() 的 速度 要 快 得 多 。 如 必须 画 一 条 音像 
素 粗 细 的 直线 ，drawLine(x,y,x,y) 的 速度 比 和 由 1]Rect(x,y,1,1) 快 。 


四 使 用 API 类 : 尽量 使 用 来 自 Java API 的 类 ， 因 为 它们 本 身 己 针 对 机 器 的 
性 能 进行 了 优化 。 这 是 用 Java 难 于 达到 的 。 比 如 在 复制 任意 长 度 的 一 个 
数组 时 ，arraryCopy0O 比 使 用 循环 的 速度 快 得 多 。 

国 蔡 换 API 类 : 有 些 时 候 ，API 类 提供 了 比 我 们 希望 更 多 的 功能 ， 相 应 的 
执行 时 间 也 会 增加 。 因 此 ， 可 定做 特别 的 版 本 ， 让 它 做 更 少 的 事情 ， 但 
可 更 快 地 运行 。 例 如 ， 假 定 一 个 应 用 程序 需要 一 个 容器 来 保存 大 量 数 
组 。 为 加 快 执 行 速 度 ， 可 将 原来 的 Vector( 矢 量 ) 替换 成 更 快 的 动态 对 
RBA o 

1. 其 他 建议 


四 将 重复 的 常数 计算 移 至 关键 循环 之 外 一 一 比如 计算 固定 长 度 缓冲 区 的 
buffer.length. 


mstatic final (静态 最 终 ) BY BCA BNF oa PE aS ILE o 
四 实现 固定 长 度 的 循环 。 

















加 使 用 javac 的 优化 选项 : -O。 它 通过 内 髓 static，final 以 及 private 方 法 ， 

从 而 优化 编译 过 的 代码 。 注 意 类 的 长 度 可 能 会 增加 (只 对 JDK ”1.1 而 言 
一 一 更 早 的 版 本 也 许 不 能 执行 字 节 查证 )。 新 型 的 “Just-in-time”(JIT) 
编译 器 会 动态 加 速 代码 。 

田 尽 可 能 地 将 计数 减 至 0 一 一 这 使 用 了 一 个 特殊 的 JVM 字 闻 人 码 。 
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“Java 优 化 工具 ”主页 : 
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PY SRE 关于 垃圾 收集 的 一 些 话 
“很 难 相信 Java 居 然 能 和 C++ 一 样 快 ， 甚 至 还 能 更 快 一 些 。” 


据 我 自己 的 实践 ， 这 种 说 法 确实 成 立 。 然 而 ， 我 也 发 现 许多 关于 速度 的 
怀疑 者 来自 一 些 早 期 的 实现 方式 。 由 于 这 些 方 式 并 非特 别 有 效 ， 所 以 没 
有 一 个 模型 可 供 参考 ， 不 能 解释 Java 速 度 快 的 原因 。 


我 之 所 以 想到 速度 ， 部 分 原因 是 由 于 C++ 模 型 。C++ 将 自己 的 主要 精力 
放 在 编译 期 间 “ 静 态 ” 发 生 的 所 有 事情 上 ， 所 以 程序 的 运行 期 版 本 非常 短 
小 和 快速 。C++ 也 直接 建立 在 C 模 型 的 基础 上 《主要 为 了 回 后 兼容 ) ， 

但 有 时 仅仅 由 于 它 在 C 中 能 按 特 定 的 方式 工作 ， 所 以 也 是 C++ 中 最 方便 
的 一 种 方法 。 最 重要 的 一 种 情况 是 C 和 C++ 对 内 存 的 管理 方式 ， 它 是 某 
些 人 党 得 Java 速 度 肯 定 慢 的 重要 依据 : 在 Java 中 ， 所 有 对 象 都 必须 在 内 
存 “ 堆 ”里 创建 。 


而 在 C++ 中 ， 对 象 是 在 堆栈 中 创建 的 。 这 样 可 达到 更 快 的 速度 ， 因 为 当 
我 们 进入 一 个 特定 的 作用 域 时 ， 堆 栈 指针 会 癌 下 移动 一 个 单位 ， 为 那个 
作用 域内 创建 的 、 以 堆栈 为 基础 的 所 有 对 象 分 配 存 储 空 间 。 而 当 我 们 离 
开 作 用 域 的 时 候 〈 调 用 完毕 所 有 局 部 构建 器 后 ) ， 堆 栈 指针 会 向 上 移动 
一 个 单位 。 然 而 ， 在 C++ 里 创建 “内 存 推 ”(Heap) 对 象 通常 会 慢 得 多 ， 

因为 它 建 立 在 C 的 内 存 堆 基础 上 。 这 种 内 存 堆 实际 是 一 个 大 的 内 存 池 ， 

要 求 必 须 进 行 再 循环 〈 再 生 ) 。 在 C++ 里 调用 delete 以 后 ， 释 放 的 内 存 会 
在 堆 里 留 下 一 个 洞 ， 所 以 再 调用 new 的 时 候 ， 存 储 分 配 机 制 必须 进行 某 
种 形式 的 搜索 ， 使 对 象 的 存储 与 堆 内 任何 现成 的 洞 相配 ， 否 则 就 会 很 快 
用 光 堆 的 存储 空间 。 之 所 以 内 存 堆 的 分 配 会 在 C++ 里 对 性 能 造成 如 此 重 
大 的 性 能 影响 ， 对 可 用 内 存 的 搜索 正 是 一 个 重要 的 原因 。 所 以 创建 基于 
堆栈 的 对 象 要 快 得 多 。 


同样 地 ， 由 于 C++ 如 此 多 的 工作 都 在 编译 期 间 进行 ， 所 以 必须 考虑 这 方 
面 的 因 系 。 但 在 Java 的 茶 些 地 方 ， 事 情 的 友 生 却 要 显得 “动态 ”得 多 ， 它 
会 改变 模型 。 创 建 对 象 的 时 候 ， 垃 圾 收集 占 的 使 用 对 于 提高 对 象 创建 的 
速度 产生 了 显著 的 影响 。 从 表面 上 看 ， 这 种 说 法 似乎 有 些 奇 怪 一 一 存储 
空间 的 释放 会 对 存储 空间 的 分 配 造 成 影响 ,但 它 正 是 JVM 采 取 的 重要 手 
段 之 一 ， 这 童 味 着 在 Java 中 为 堆 对 象 分 配 存 储 空间 几乎 能 达到 与 C++ 中 
在 堆栈 里 创建 存储 空间 一 样 快 的 速度 。 



































可 将 C++ 的 堆 《〈 以 及 更 慢 的 Java 堆 ) 想象 成 一 个 隆 院 ， 每 个 对 象 都 拥有 
目 己 的 一 块 地 皮 。 在 以 后 的 某 个 时 间 ， 这 种 “不 动产 ”会 被 抛弃 ， 而 且 必 
须 再 生 。 但 在 某 些 JVM 里 ，Java 堆 的 工作 方式 却 是 鼎 有 不 同 的 。 它 更 象 
一 条 传送 带 : 每 次 分 配 了 一 个 新 对 象 后 ， 都 会 朝 前 移动 。 这 意味 看 对 象 
存储 空间 的 分 配 可 以 达到 非常 快 的 速度 。“ 堆 指针 ”简单 地 向 前 移 至 处 女 
地 ， 所 以 它 与 C++ 的 堆栈 分 配方 式 几乎 是 完全 相同 的 (当然 ， 在 数据 记 
录 上 会 多 花 一 些 开销 ， 但 要 比 搜索 存储 空间 快 多 了 ) 。 


现在 ， 大 家 可 能 注意 到 了 堆 事 实 并 非 一 条 传送 高 。 如 按 那 种 方式 对 答 
它 ， 最 终 就 要 求 进行 大 量 的 页 交换 《这 对 性 能 的 发 挥 会 产生 巨大 干 
扰 ) ， 这 样 终究 会 用 光 内 存 ， 出 现 内 存 分 页 错误 。 所 以 这 儿 必 须 采 取 一 
个 技巧 ， 那 就 是 著名 的 “垃圾 收集 器 ”。 它 在 收集 “垃圾 ”的 同时 ， 也 负责 
压缩 堆 里 的 所 有 对 象 ， 将 “ 堆 指 针 ” 移 至 尺 可 能 靠近 传送 带 开头 的 地 方 ， 
LARE (AF) 分 页 错误 的 地 点 。 垃 圾 收集 器 会 重新 安排 所 有 东西 ， 
使 其 成 为 一 个 高 速 、 无 限 上 自由 的 堆 模 型 ， 同 时 游 思 有 余地 分 配 存储 衬 
间 。 


为 真正 掌握 它 的 工作 原理 ， 我 们 首先 需要 理解 不 同 垃圾 收集 器 CGC) 

采取 的 工作 方案 。 一 种 简单 、 但 速度 较 慢 的 GC 技术 是 引用 计数 。 这 意 
味 着 每 个 对 象 都 包含 了 一 个 引用 计数 器 。 每 当 一 个 句柄 同一 个 对 象 连 接 
起 来 时 ， 引 用 计数 器 就 会 增值 。 每 当 一 个 句柄 超出 自己 的 作用 域 ， 或 者 
设 为 null 时 ， 引 用 计数 就 会 减 值 。 这 样 一 来 ， 只 要 程序 处 于 运行 状态 ， 

就 需要 连续 进行 引用 计数 管理 一 一 尽管 这 种 管理 本 身 的 开销 比较 少 。 垃 
圾 收集 器 会 在 整个 对 象 列 表 中 移动 巡视 ， 一 旦 它 发 现 其 中 一 个 引用 计数 
成 为 0， 就 释放 它 占据 的 存储 空间 。 但 这 样 做 也 有 一 个 缺点 : 若 对 象 相 
互 之 间 进 行 循环 引用 ， 那 么 即使 引用 计数 不 是 0， 仍 有 可 能 属于 应 收 掉 
的 “垃圾 >”。 为 了 找 出 这 种 自 引 用 的 组 ， 要 求 垃圾 收集 器 进行 大 量 额外 的 
工作 。 引 用 计数 属于 垃圾 收集 的 一 种 类 型 ， 但 它 看 起 来 并 不 适合 在 所 有 
JVM 方 案 中 采用 。 


在 速度 更 快 的 方案 里 ， 垃 圾 收集 并 不 建立 在 引用 计数 的 基础 上 。 相 友 ， 
它们 基于 这 样 一 个 原理 : 所 有 非 死 锁 的 对 象 最 终 都 衣 定 能 回调 至 一 个 句 
柄 ， 该 句柄 要 么 存在 于 堆栈 中 ， 要 么 存在 于 静态 存储 空间 。 这 个 回调 链 
可 能 经 历 了 几 层 对 象 。 所 以 ， 如 果 从 堆栈 和 静态 存储 区 域 开 始 ， 并 经 历 
所 有 句柄 ， 就 能 找 出 所 有 活动 的 对 象 。 对 于 自己 找到 的 每 个 句柄 ， 都 必 
须 跟 躁 到 它 指向 的 那个 对 象 ， 然 后 跟随 那个 对 象 中 的 所 有 人 句柄 ,，“ 跟 中 
退 击 ?到 它们 指 同 的 对 象 .……. 等 等 ， 直 到 壳 历 了 从 堆栈 或 静态 存储 区 域 




















中 的 句柄 发 起 的 整个 链接 网 路 为 止 。 中 途 移 经 的 每 个 对 象 都 必须 仍 处 于 
活动 状态 。 注 意 对 于 那些 特殊 的 目 引 用 组 ， 并 不 会 出 现 前 述 的 问题 。 由 
于 它们 根本 找 不 到 ， 所 以 会 自动 当 作 垃圾 处 理 。 


在 这 里 阐述 的 方法 中 ，JVM 采 用 一 种 “ 目 适 应 ”的 垃圾 收集 方案 。 对 于 它 
找到 的 那些 活动 对 象 ， 具 体 采取 的 操作 取决 于 当前 正在 使 用 的 是 什么 变 
体 。 其 中 一 个 变 体 是 “停止 和 复制 *。 这 意味 着 由 于 一 些 不 久之 后 就 会 非 
常 明显 的 原因 ， 程 序 首 先 会 停止 运行 (并非 一 种 后 台 收 集 方案 〉。 随 

后 ， 已 找到 的 每 个 活动 对 象 都 会 从 一 个 和 内存 扒 复制 到 另 一 个 ， 留 下 所 有 
的 垃圾 。 除 此 以 外 ， 随 奢 对 象 复制 到 新 堆 ， 它 们 会 一 个 接 一 个 地 聚焦 在 
一 起 。 这 样 可 使 新 堆 显 得 更 加 紧凑 (并 使 新 的 存储 区 域 可 以 简单 地 抽 离 
末尾 ， 就 象 前 面 讲述 的 那样 ) 。 


当然 ， 将 一 个 对 象 从 一 处 挪 到 另 一 处 时 ， 指 回 那 个 对 象 的 所 有 句柄 〈 引 
HO 都 必须 改变 。 对 于 那些 通过 跟踪 内 存 扒 的 对 象 而 获得 的 句柄 ， 以 及 
那些 静态 存储 区 域 ， 都 可 以 立即 改变 。 但 在 “ 志 历 ?过程 中 ， 还 有 可 能 遇 
到 指向 这 个 对 象 的 其 他 句柄 。 一 旦 发 现 这 个 问题 ， 束 当即 进行 修正 (可 
想象 一 个 散 列表 将 老 地 址 映射 成 新 地 址 〉。 


有 两 方面 的 问题 使 复制 收集 器 显得 效率 低下 。 第 一 个 问题 是 我 们 拥有 两 
个 堆 ， 所 有 内 存 都 在 这 两 个 独立 的 堆 内 来 回 移动 ， 要 求 付 出 的 管理 量 是 
实际 需要 的 两 倍 。 为 解决 这 个 问题 ， 有 些 JVM 根 据 需 要 分 配 内 存 堆 ， 并 
将 一 个 扒 简 单 地 复制 到 另 一 个 。 


第 二 个 问题 是 复制 。 随 独 程 序 变 得 越 来 越 “ 健 壮 >， 它 几乎 不 产生 或 产生 
很 少 的 垃圾 。 尽 管 如 此 ， 一 个 副本 收集 器 仍 会 将 所 有 内 存 从 一 处 复制 到 
为 一 处 ， 这 显得 非常 浪费。 为 避免 这 个 问题 ， 有 些 JVM 能 侦 测 是 否 没 有 
产生 新 的 垃圾 ， 并 随即 改换 另 一 种 方案 〈 这 便 是 “ 目 适 应 ”的 缘由 ) o A 
一 种 方案 叫 作 “标记 和 清除 ”，Sun 公 司 的 JVM 一 直 采 用 的 都 是 这 种 方 
和 案 。 对 于 钊 规 性 的 应 用 ， 标 记 和 清除 显得 非常 慢 ， 但 一 旦 知道 目 己 不 产 
生 垃 圾 ， 或 者 只 产生 很 少 的 垃圾 ， 它 的 速度 就 会 非常 快 。 


标记 和 清除 采用 相同 的 逻辑 : 从 堆栈 和 静态 存储 区 域 开 始 ， 并 跟 踩 所 有 
句柄 ， 寻 找 活动 对 象 。 然 而 ， 每 次 及 现 一 个 活动 对 象 的 时 候 ， 残 会 设置 
一 个 标记 ， 为 那个 对 象 作 上 “记号 ”。 但 此 时 尚 不 收集 那个 对 象 。 只 有 在 
标记 过 程 结束 ， 清 除 过 程 才 正 式 开始 。 在 清除 过 程 中 ， 死 锁 的 对 象 会 被 
释放 然而 ， 不 会 进行 任何 形式 的 复制 ， 所 以 假 耕 收集 占 决 定 压缩 一 个 断 
续 的 内 存 堆 ， 它 通过 移动 周围 的 对 象 来 实现 。 


















































“停止 和 复制 ?向 我 们 表明 这 种 类 型 的 垃圾 收集 并 不 是 在 后 台 进 行 的 ， 相 
有 反 ， 一 旦 发 生 垃 圾 收集 ， 程 序 束 会 停止 运行 。 在 Sun 公 司 的 文档 库 中 ， 

可 发 现 许 多 地 方 都 将 垃圾 收集 定义 成 一 种 低 优 先 级 的 后 人 台 进 程 ， 但 它 只 
是 一 种 理论 上 的 实验 ， 实 际 根本 不 能 工作 。 在 实际 应 用 中 ，Sun 的 垃圾 
收集 右 会 在 内 存 减少 时 运行 。 除 此 以 外 ,， “标记 和 清除 ?也 要 求 程 序 停止 
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运行 。 


正如 早先 指出 的 那样 ， 在 这 里 介绍 的 JVM 中 ， 内 存 是 按 大 块 分 配 的 。 若 
分 配 一 个 大 块头 对 象 ， 它 会 获得 目 己 的 内 存 块 。 严 格 的 “停止 和 复制 ?要 
求 在 释放 旧 扒 之 前 ， 将 每 个 活动 的 对 象 从 源 堆 复制 到 一 个 新 堆 ， 此 时 会 
WRB AER LYE GHIA IR, HER ae By BY Al) A IR I 
制 对 象 ， 就 象 它 进行 收集 时 那样 。 每 个 块 都 有 一 个 生成 计数 ， 用 于 跟踪 
它 是 否 依然 “存活 ?。 通 向， 只 有 目 上 次 垃圾 收集 以 来 创建 的 块 才 会 得 到 
压缩 ;对 于 其 他 所 有 块 ， 如 果 已 从 其 他 东 些 地 方 进行 了 引用 ， 那 么 生成 
计数 部 会 注 出 。 这 古诗 多 短期 的 、 临 时 的 对 象 经 党 过 到 的 情况 。 会 周期 
性 地 进行 一 次 完整 清除 工作 一 一 大 块 尖 的 对 象 仍 示 复制 (只 是 让 它们 的 
生成 计数 洪 出 ) ， 而 那些 包含 了 小 对 象 的 块 会 进行 复制 和 压 纵 。JVM 会 
监视 垃圾 收集 器 的 效率 ， 如 果 由 于 所 有 对 象 都 属于 长 期 对 象 ， 造 成 垃 专 
收集 成 为 浪费 时 间 的 一 个 过 程 ， 束 会 切换 到 “标记 和 清除 ”方案 。 类 似 

地 ，JVM 会 跟 踊 监 视 成 功 的 “标记 与 清除 ”工作 ， 辱 内 存 堆 变 得 越 来 

越 “散乱 ?， 残 会 换 回 “ 停 止 和 复制 * 方 案 。“ 自 定义 ”的 说 法 就 是 从 这 种 行 
为 来 的 ， 我 们 将 其 最 后 总 结 为 :“ 根 据 情况 ， 目 动 转换 停止 和 复制 / 标 
记 和 清除 这 两 种 模式 ”。 


JVM 还 采用 了 其 他 许多 加 速 方案 。 其 中 一 个 特别 重要 的 涉及 闭 载 器 以 及 
JIT SR PEAS oA AUR PSS 〈 通 向 是 我 们 首次 想 创 建 那个 类 的 一 个 
MAN) ， 会 找到 .class 文 件 ， 并 将 那个 类 的 字 市 码 送 入 内 存 。 此 时 ， 一 
个 方法 是 用 JIT 编 译 所 有 代码 ， 但 这 样 做 有 两 方面 的 缺点 它 会 化 更 多 
的 时 间 ， 辱 与 程序 的 运行 时 间 综 合 考 虑 ， 编 译 时 间 还 有 可 能 更 长 ;而 且 
它 增 大 了 执行 文件 的 长 度 〈 字 节 码 比 扩展 过 的 JII 代 码 精 简 得 多 ) ， 这 
有 可 能 造成 内 存 页 交换 ， 从 而 显著 放 慢 一 个 程序 的 执行 速度 。 另 一 种 答 
代办 法 是 : 除非 确 有 必要 ， 人 否则 不 经 JIT 编 译 。 这 样 一 来 ， 那 些 根本 不 
会 执行 的 代码 就 可 能 永远 得 不 到 JIT 的 编译 。 


由 于 JVM 对 浏览 右 来 说 是 外 置 的 ， 大 家 可 能 和 希望 在 使 用 浏览 右 的 时 候 从 
一 些 JVM 的 速度 提高 中 获得 好 处 。 但 非常 不 乎 ，JVM 目 前 不 能 与 不 同 的 
浏览 器 进行 沟通 。 为 发 挥 一 种 特定 JVM 的 潜力 ， 要 么 使 用 内 建 了 那 种 



































JVM 的 浏览 器 ， 要 么 只 有 运行 独立 的 Java 应 用 程序 。 


本 书 由 “ 行 行 ?整理 ， 如 果 你 不 知道 读 什 么 书 或 者 想 获得 更 多 免费 电子 书 
请 加 小 编 微 信 或 QQ: 2338856113 小 编 也 和 结交 一 些 喜 欢 读书 的 朋友 或 
者 关注 小 编 个 人 微 信 公众 号 名 称 : 幸福 的 味道 id: d716-716 为 了 方便 书 
友 朋 友 找 书 和 看 书 ， 小 编 自己 做 了 一 个 电子 书 下 载 网 站 ， 网 站 的 名 称 
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